本文对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次。