java的字符串存储在堆中还是常量池中
一、给一个字符串变量赋值有两种方式
- String str1=new String("This is a string");
- String str2 ="This is a string";
第一种方式通过关键字new定义:编译程序先在字符串常量池查找,是否存在"This is a string"常量,如果不存在,则在字符串常量池开辟一个内存空间,存放"This is a string";如果存在,则不另外开辟空间,保证字符串常量区只有一个"This is a string",节省空间。然后在堆区,开辟一个空间,存放new出来的String对象,并在栈区开辟空间,存放变量名称str1,str1指向堆区new出来的String对象。
第二种方式直接定义:在字符串常量区查找是否存在"This is a string"常量,如果不存在,则在字符串常量区开辟一个内存空间,存放"This is a string";如果存在,则不另外开辟空间;在栈区开辟空间,存放变量名称str2,str2指向字符串常量池"This is a string"的内存地址。
二、String的intern方法使用
当调用intern()方法时,如果字符串常量池中存在该字符串,则返回池中该字符串的引用;否则,将此字符串添加到字符串常量池,并返回字符串的引用。
jdk1.6与jdk1.7处理字符串的相同点:
如果字符串常量池中存在,则返回池中字符串的引用。
jdk1.6与jdk1.7处理字符串的不同点:
如果字符串常量池中不存在该字符串,jdk1.6会将该字符串拷贝到字符串常量池中,而jdk1.7会在字符串常量池中生成该字符串实例的引用。
String str1=new StringBuilder("this is ").append("a string").toString();
//执行完这行代码后,常量池中会有"this is "和"a string",但是不会有"this is a string"。
System.out.println(str1.intern()==str1);//true
String str2=new String("this is a string");
//执行完这行代码后,常量池中会有"this is a string"。
System.out.println(str2==str2.intern());//false
三、字符串常量池在JVM数据区的演进
这里顺带提下常量池与运行时常量池的区别:
常量池:也叫静态常量池,在编译后的class文件中,分为字面量和符号引用量,还没有加载到虚拟机JVM中;
运行时常量池:JVM加载class文件,并为静态常量池分配内存空间,此时的常量池即为运行时常量池。除了编译后产生的常量,这个常量池在运行时,还会增加新的常量:intern()方法生成的新的字符串常量、由基础数据生成的包装类型。