Java 6 和 7 中String.intern返回引用的不同
先看看下面这段代码,这段代码在Java 6 和Java 7中的运行的结果并不同,为什么呢?
public class Main {
public static void main(String[] args) {
String myString = "Rafe";
String str1 = new StringBuilder("好好").append("学习").toString();
System.out.println(str1.intern() == str1);
String str2 = new StringBuilder("Ra").append("fe").toString();
System.out.println(str2.intern() == str2);
}
}
在Java 6 中运行,会得到两个false。而在Java 7中运行,会得到一个true和一个false。
先分析第一段代码。原因在于,Java6中字符串常量池是在永久代区。intern方法会将首次遇到的字符串复制到那里,返回的也是永久代中这个字符串的引用。而通过str1本身是通过StringBuilder构造出来的,是堆内存中的对象。这两个对象势必不同,因此返回false。在Java 7 中,字符串常量池被放到了堆中,intern的实现也不再是复制实例,只是在常量池中记录首次出现的实例引用。此时intern返回的引用和StringBuilder创建的堆对象实则为同者了。
在看第二段代码。它在两个不同版本的Java中结果相同。“Rafe”在此前已经出现过并记录到了常量池中。而str2的是另外一个由StringBuilder创建的堆对象,只是内容同之前缓存的字符串。无论是java 6中返回的常量池字符串,还是Java 7返回的常量池中记录的堆对象引用(此引用实则为myString),都与str2不同,因此输出false。