jdk1.6中字符串常量池存放在永久代中;jdk1.7中字符串常量池存放在堆中。
String的intern()方法:
①jdk1.6及之前的版本中只能查询或创建在字符串常量池;
②jdk1.7及以后的版本中会先查询字符串常量池,若没有则会到堆中查询并存储堆的引用,然后返回。(参考博客)
下面从 JDK6 和 JDK7 分析在实现上的区别(引号中声明的字符串都会放在字符串常量池中,new出的字符串都会放到堆中)
以上代码在 JDK6 中的结果是 false false
在 JDK7 中的结果是 false true
JDK6
- 对于第一部分代码
"a"被放在字符串常量池中,然后(因为new)存放在堆中,s指向堆中的"a";
[s.intern()]当把"a"副本放入字符串常量池时,发现有则不再放置,后面s2="a"则s2指向字符串常量池中的"a",因此s != s2 - 对于第二部分代码
"a"被放在字符串常量池中,后来两个new出来的"a"相加,new出"aa"然后放在了堆中,s3指向堆中的"aa";
[s3.intern()]当把"aa"副本放在字符串常量池中时,发现字符串常量池没有"aa",则放入(由于是副本则地址与堆中的不一样)。s4="aa"则s4指向字符串常量池中的"aa",因此s3 != s4
JDK7
- 对于第一部分代码
同上第一部分代码解释(副本改为引用) - 对于第二部分代码
"a"被放在字符串常量池中,后面new出来的"a"相加,可以(隐含)new出"aa"然后放在堆中,s3指向堆中的"aa";
[s3.intern()]当把"aa"引用放在字符串常量池中时,发现没有则存入它的引用(此时与堆中的地址一样)。s4="aa"则s4指向字符串常量池中的"aa"(相当于指向堆中的"aa"),因此s3 == s4