正经解释:
String.intern()是一个Native方法,它的作用是:如果字符串常量池中已经包含一个等于此String对象的字符串,则返回代表池中这个字符串的String对象;否则,将此String对象包含的字符串添加到常量池中,并且返回此String对象的引用。
应该一下看不太明白,那就看下面的这段代码。
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
public static void main(String[] args) {
String str1 = new StringBuilder("计算机").append("软件").toString();
System.out.println(str1.intern() == str1);
String str2 = new StringBuilder("ja").append("va").toString();
System.out.println(str2.intern() == str2);
} }
}
这段代码在JDK 1.6中运行,会得到两个false,而在JDK 1.7中运行,会得到一个true和一个false。产生差异的原因是:在JDK 1.6中,intern()方法会把首次遇到的字符串实例复制到永久代中,返回的也是永久代中这个字符串实例的引用,而由StringBuilder创建的字符串实例在Java堆上,所以必然不是同一个引用,将返回false。而JDK 1.7(以及部分其他虚拟机,例如JRockit)的intern()实现不会再复制实例,只是在常量池中记录首次出现的实例引用,因此intern()返回的引用和由StringBuilder创建的那个字符串实例是同一个。对str2比较返回false是因为“java”这个字符串在执行StringBuilder.toString()之前已经出现过,字符串常量池中已经有它的引用了,不符合“首次出现”的原则,而“计算机软件”这个字符串则是首次出现的,因此返回true。
也就是字符串初次出现,则在字符串常量池中保存对该实例的引用。
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
String str1 = new StringBuilder("计算机").append("软件").toString();
System.out.println(str1);
System.out.println(str1.intern() == str1);
//“计算机软件”这个字符串则是首次出现的,因此返回true.也即是说常量池中关于字符串的引用和在堆中的引用是一样的。
String ss=new String("计算机软件");
System.out.println(ss.intern() == ss);
//已经不是第一次出现了,在堆中新建实例变量ss,而常量池中字面量"计算机软件"的引用依然是创建时保存的,也即是 System.out.println(str1 == ss);结果当然是false;
String so="计算机软件";
System.out.println(so == str1);
//so没有使用new进行创建,直接查看常量池中是否已经存在字面量"计算机软件",因为已经存在,因此直接将该字面量的引用赋给了so,所以so和str1是一个引用。
System.out.println(so == ss);
}
}
结果:
计算机软件
true
false
true
false