通过参考《深入理解java虚拟机》和几篇博客总结并实践得出来的经验
JDK1.6和JDK1.7之间intern方法的实现发生了一些变化
JDK各版本String的共同之处:
用引号做声明的字符串都会在常量池中生成,而new出来的字符串对象则是在堆中生成的,所以它们两个的内存地址肯定是不同的。
所以,String str = new String("abc");语句其实可以分为两步 :
1.在常量池中生成"abc"字符串。
2.在堆中生成String对象,并把此对象的引用赋给str。
因此可以得出(str == "abc")为false,本人做过实践,结果也和预期一致。
JDK1.6及其以下版本的intern方法:
首先1.6及其以下版本的常量池位于方法区,假设str = new String("abc"); 那么str.intern()的作用是先查看常量池里是否有"abc"这个字符串对象,如果有,则返回常量池里对象"abc"的引用,否则会先把"abc"字符串实例复制到常量池中,然后再返回该实例的引用。
栗子:
//1.在常量池中生成"abc"字符串。
//2.在堆中生成String对象,并把此对象的引用赋给str1。
String str1 = new String("abc");
//"abc"对象已经在常量池里,所以这里intern只用来返回"abc"对象的引用
str1.intern();
//常量池里已经存在"abc"对象,所以返回"abc"对象的引用
String str2 = "abc";
//注意,str1指向的是堆里的String对象,而不是指向常量池的"abc",而str2指向的就是常量池里的"abc"
System.out.println(str1 == str2);
//1.在常量池中分别生成"def"、"ghi"字符串。
//2.在堆中生成2个String对象,并把这两个对象相加生成的新对象的引用赋给str3。
String str3 = new String("def") + new String("ghi");
//常量池里不存在"defghi"对象,所以先复制"defghi"对象到常量池,然后返回"defghi"对象的引用
str3.intern();
//上一步常量池已经存放了"defghi"对象,所以str4引用的是常量池对象
String str4 = "defghi";
System.out.println(str3 == str4);
运行结果:
false
false
JDK1.7及其以上版本的intern方法:
首先1.7及其以上版本的常量池位于堆,假设str = new String("abc"); 那么str.intern()的作用是先查看常量池里是否有"abc"这个字符串的引用,如果有,则返回常量池里对象"abc"的引用,否则会先把"abc"字符串的引用复制到常量池中,然后再返回该引用。
栗子:
//1.在常量池中生成"abc"字符串的引用。
//2.在堆中生成String对象,并把此对象的引用赋给str1。
String str1 = new String("abc");
//"abc"对象的引用已经在常量池里,所以这里intern只用来返回"abc"对象的引用
str1.intern();
//常量池里已经存在"abc"对象,所以返回"abc"对象的引用
String str2 = "abc";
//注意,str1指向的是堆里的String对象,而不是指向常量池的"abc",而str2指向的就是常量池里的"abc"
System.out.println(str1 == str2);
//1.在常量池中分别生成"def"、"ghi"字符串的引用。
//2.在堆中生成2个String对象,并把这两个对象相加生成的新对象的引用赋给str3。
String str3 = new String("def") + new String("ghi");
//常量池里不存在str3(str3是引用类型),所以先复制str3到常量池,然后返回str3的值
str3.intern();
//上一步常量池已经存放了str3,所以str4的值是常量池引用(也就是str3)
String str4 = "defghi";
System.out.println(str3 == str4);
运行结果:
false
true