参考地址:class常量池、字符串常量池和运行时常量池的区别_那年那些事儿-CSDN博客_运行时常量池和字符串常量池的区别
public static void main(String[] args) {
String s1="hello";
String s2="hello";
String s3=new String("world");
String s4=s3.intern();
String s5="world";
String s6="helloworld";
String s7=s1+s5;
System.out.println(s1 == s2);//true
System.out.println(s3 == s4);//false
System.out.println(s4 == s5);//true
System.out.println(s6 == s7);//false
}
解析程序的内存分配过程:
1.首先,解析第一句的时候,在堆中会有hello的实例对象(驻留字符串对象),全局StringTable(在 HotSpot VM 里实现的 String Pool 功能的是一个 StringTable 类,它是一个哈希表)中存放"hello"的一个引用值。
2.与第一句是不同的实例,当在解析第二句的时候查找StringTable,里面有hello的全局驻留字符串的引用,所以s2的引用地址与s1的引用地址相同。所以是s1=s2.
3.然后解析到第三句,堆中会生成两个实例对象,一个是world的实例对象(驻留字符串对象),并且在StringTable中存储一个world的引用值,另外一个是new出来的world的实例对象,返回对象的引用。整个过程可以看做是以下步骤的:String tmp="world";String s3=new String(tmp);
4.解析到第四句,调用s3的intern函数,返回StringTable中world的引用值,如果StringTable中没有,就把s3的引用值添加进去,这里已经有了,所以返回tmp中world的引用值。所以s3!=s4。
5.解析到第五句,返回存在StringTable中world的引用值。所以s4=s5。
6.解析到第六句,过程同第1步,在堆中会有helloworld的主流字符串对象,并在StringTable中会存放它的引用。
7.解析第七句,在s1+s5的过程,jdk1.5之后编译器有个优化,会自动把"+"替换成StringBulider的append,先用StringBuilder stb=new StringBuilder(s1的驻留字符串对象)生成一个StringBuilder对象,stb.append(s5的驻留字符串对象),接着调用StringBuilder的toString()方法在堆中创建一个String对象,最后返回生成的String对象的地址引用。所以s6!=s7