问题源于上课时,对String str = "abc"的疑问。本来是老师给我们介绍了一道经典面试题,题目如下:
String str01 = "abc";
String str02 = "abc";
String str03 = new String("abc");
String str04 = new String("abc");
System.out.println(str01 == str02); //true
System.out.println(str03 == str04); //false
System.out.println(str01.equals(str02)); //true
System.out.println(str03.equals(str04)); //true
对于“==”和equals(),没有疑问。
就是不清楚 String str01 = "abc";中abc,为何是存储在常量池中。系统难道隐式给加了final修饰符,str01不是变量,是常量?
为了搞清这个问题,我查了java对内存的分类。
(网上资料)程序中用来存放数据的内存分为四块,另有一块用于存放代码
1.堆:存放所有new出来的对象(我们知道java并没有全局变量这个概念,有人是把它单独放在properties文件使用,充当全局变量的作用.)
2.栈:存放基本类型的变量数据和对象的引用。对象引用即指向对象值的地址存在栈中,而对象值则存放在堆中或者常量池中(字符串常量对象存放在常量池中)。
3.常量池:存放基本类型常量和字符串常量。
栈和常量池中的对象可以共享,堆中的对象不可以共享。
对于字符串来说,其对象的引用都是存储在栈中,如果是编译期就已建好的(没有new)就存在常量池中,如果是运行期(有new)则存在堆中。
解释:String str03 = new String("abc");(如下图所示)
通过new操作产生一个字符串“abc”时,会先去常量池中查找是否有"abc"对象;如果没有,则在常量池中创建一个此字符串对象,然后堆中在创建一个常量池中此“abc”对象的拷贝对象。
至此,问题的答案显而易见。无论是new还是直接赋值,"abc"都作为字符串常量存在常量池中,只是new多了一层引用。
回想一下,八大基本类型与对象的转换,除了Interger和character外,其余都只是首字母大写即为对应的对象。
String在这不是字符串类型,是字符串对象的引用类型。引用类型存在栈中,值存在常量池中。
Plus:栈的优势:存取速度比堆快,仅次于寄存器;栈数据可共享。 缺点:存在栈中的数据大小和生存期必须是确定的,缺乏灵活性。
堆的优势:运行时动态分配内存。 缺点:由于要在运行时动态分配内存,存取速度较慢。