下面是 String.intern() 方法的 Javadoc 注释:
*返回字符串对象的规范表示。
字符串池,最初为空,由 String 类私下维护。
当调用 intern 方法时,如果池中已经包含一个等于该 String 对象的字符串,该字符串由 equals(Object) 方法确定,则返回池中的字符串。否则,将此 String 对象添加到池中并返回对该 String 对象的引用。
由此可见,对于任何两个字符串 s 和 t,当且仅当 s.equals(t) 为真时,s.intern() == t.intern() 才为真。
所有文字字符串和字符串值的常量表达式都是实习的。字符串文字在 Java™ 语言规范的第 3.10.5 节中定义。”
但我认为从 jdk-8u102 开始发生了一些变化。
检查以下示例:
public class Test1 {
public static void main(String[] args) {
String s1 = new String(new char[]{'J', 'a', 'v', 'a'});
String s2 = s1.intern();
System.out.println(s1 == s2);
}
}
如果您在 JDK 7u80(JDK 7 的最新稳定版本)和 JDK 8 到 8u101 中运行上述程序,则输出为:
true
但是,如果您在 JDK 8u102 及 JDK 9 和 JDK 10 中运行上述程序,则输出为:
false
为什么 intern() 方法从 JDK 8u102 开始表现不同?
我检查了发行说明和 Javadoc 注释,但找不到任何与 JDK 8u102 中的 intern() 方法相关的更改。
我检查了博客和其他网站,但没有运气。
但是当我尝试使用其他字符串时,输出没有变化:
public class Test2 {
public static void main(String[] args) {
String s3 = new String(new char[]{'U', 'd', 'a', 'y', 'a', 'n'});
String s4 = s3.intern();
System.out.println(s3 == s4);
}
}
上面的程序在 JDK 7、JDK 8、JDK 9 和 JDK 10 中总是打印true 。
只有在加载 Test1 类之前字符串池表引用了“Java”时,此行为才可能发生。
s1 引用 HEAP 上的字符串对象“Java”,s1.intern() 返回字符串池对象的引用(因为字符串池已经引用了“Java”)。
这就是 s1 == s2 返回 false 的原因。
但是当加载 Test2 类时,字符串池表不会引用“Udayan”。
s3 引用 HEAP 上的字符串对象“Udayan”,s3.intern() 将 s3 引用的字符串对象添加到字符串池并返回相同的引用。这意味着 s3 和 s4 指的是同一个对象。
这就是 s3 == s4 返回 true 的原因。
如果我的观察是正确的,那么这意味着字符串池最初不是空的。
字符串池最初包含“Java”、“java”、“Oracle”和其他字符串对象。
任何人都可以确认这一点吗?