jvm学习——String类的intern方法本质理解——丑九怪
java产生字符串的两种方式
1、String str1 = new String(“填入字符串”);
- 这种方式本质上是在jvm的堆空间中创建对象,因为str是“new”出来的,作为对象,保存在堆空间中
- 在堆中创建了对象之后,将对象的引用返回给jvm虚拟机栈的某个栈帧,由其局部变量表的reference保存起来
2、String str2 = “填入字符串”;
- 这种方式相对复杂,首先查看运行时常量池中是否存在该字符串的引用。
– 如果存在,那么直接返回运行时常量池中的引用给reference保存起来
– 如果不存在,先在java堆中创建该字符串的对象,然后将该字符串对象的引用保存在运行时常量池中,最后返回运行时常量池中的引用给reference保存起来
区别
- 第一种情况reference中的引用指向堆空间的地址
- 第二种情况reference中的引用指向常量池的地址
intern本质
- 上述例子中,str1≠str2,因为地址不同,本质上是两块内存空间
如果再执行下面代码,结果如何?
String str3 = str1.intern();
System.out.println("str1 == str2: " + (str1 == str2));
System.out.println("str2 == str3: " + (str2 == str3));
- 理解:str1.intern(); 这个方法执行时,JVM先检查字符串常量池中是否有相同字符串,如果有,就返回该字符串常量的引用地址。如果没有,则在字符串常量池中创建该字符串常量,然后返回该字符串常量的引用地址。