堆、栈、常量池的理解

发布于:2022-12-24 ⋅ 阅读:(448) ⋅ 点赞:(0)

        本文对java的栈、堆和常量池之间的关系以及存放的内容讲一些个人浅显的理解

1.栈:

        存放基本类型的数据对象的引用,但对象本身不存放在栈中,而是存放在堆中。

我的理解:栈其实是用于存放对象的地址,对象的引用其实就是地址

比如:

Test test = new Test();

        test指向的对象 的物理地址是110925,而这个地址存放的位置就在栈内存中。如果输出test那就是110925,其实就是对象的地址。

注意: 这个“地址“可以指向堆中的对象,也可以指向堆中的常量池的常量对象

2.堆:

        堆中内存,存放的是对象实例,以及常量池。

对象实例:

        我们知道,new就会产生对象,这个对象就存放在堆中,其地址存放在栈里,通过栈的地址就可以找到堆中新建的对象。

比如,对于任意CNM类:

CNM cnm = new CNM();

new出来的对象会放在堆中,地址是cnm

常量池:

        有时候,我们不需要new出新对象,而是让栈中的地址直接指向常量池常量

比如:

int a = 5;

则常量池中5的地址是a分配到的内存地址,a是这块内存的名称。

        java中的基本类型有:byte、short、char、int、long、boolean

        其对应的包装类分别是:Byte、Short、Character、Integer、Long、Boolean

        上边提到的这些包装类都实现了常量池技术,而两种浮点数类型的包装类则没有实现。另外,String类型也实现了常量池技术。

注意:

        以上提到的几种基本类型包装类均实现了常量池技术,但他们维护的常量仅仅是【-128至127】这个范围内的常量,如果常量值超过这个范围,就会从堆中创建对象,不再从常量池中取。比如,Integer i1 = 400; Integer i2 = 400;,很明显超过了127,无法从常量池获取常量,就要从堆中new新的Integer对象,这时i1和i2不相等。

例如,我们可以:

String str = new String("hello");

也可以:

String str1 = "hello";

区别:前者是new了新的对象,而后者是直接指向常量池。

        因此,str != str1;(因为其内存地址不同,但要注意,其hashcode是相同的,因为hashcode根据具体值来计算)

那么如果有:

String str2 = new String("hello");

仍然有 str2 != str

因为两次分别new了新的对象,地址是不一样的。

进阶:

String str = new String("abc")

(面试题)创建了几个对象?

答:1个或2个。

首先,在堆中new出一个对象实例,通过栈的str指向它,这是第一个对象。

然后,要看常量池中是否具有常量对象“abc”

如果有,那么new出来的对象就直接与常量池的常量对象连接起来。

如果无,那么会自动在常量池创建新的常量对象“abc”并且连接起来。

所以是1次或2次。

本文含有隐藏内容,请 开通VIP 后查看