String.intern()是一个Native方法,它的作用是:
如果字符串常量池中已经包含了一个等于此String对象的字符串,则返回代表池(运行时常量池)中这个字符串的String对象;
否则,将此String对象包含的字符串添加到常量池中并且返回此String对象的引用。
此方法在jdk1.6和jdk1.7中有差异,下面来看段代码:
String s1 = new StringBuilder("go")
.append("od").toString();
System.out.println(s1.intern() == s1);
String s2 = new StringBuilder("ja")
.append("va").toString();
System.out.println(s2.intern() == s2);
这段代码在jdk1.6中运行,会得到两个false,而在jdk1.7中运行会得到一个true一个false。
产生差异的原因是:
-
在jdk1.6中,intern()方法会把首次遇到的字符串实例复制到永久代中,返回的也是永久代中这个字符串实例的引用,而用StringBuilder创建的字符串实例在Java堆上,所以必然不是同一个引用,将返回false。
-
而jdk1.7中的intern()实现不会再复制实例,只是在常量池中记录首次出现的实例引用,因此intern()返回的引用和由StringBuilder创建的那个字符串实例是同一个。
-
对str2比较返回false是因为“java”这个字符串在执行StringBuilder.toString()之前已经出现过,字符串常量池中已经有它的引用了,不符合首次出现的原则,而“good”这个字符串则是首次出现的,因此返回true。