正如 String.intern() 方法的 Javadoc 中提到的那样,字符串池最初真的是空的吗?

Is String Pool really empty initially as mentioned in the Javadoc of String.intern() method?

下面是 JavaString.intern() 方法的文档注释:

*Returns 字符串对象的规范表示。

一个字符串池,最初是空的,由 class 字符串私下维护。

调用 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 的最后一个 table 版本)和 JDK 8 中 运行 以上程序直到 8u101,然后输出为:
正确

但是如果你运行上面的程序在JDK8u102之后和JDK9&JDK10,那么输出是:

为什么 intern() 方法开始表现不同 JDK 8u102 之后?

我检查了发行说明和 Java文档评论,但找不到任何关于 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 和 [=94 中打印 true =] 10.

仅当“Java”在加载 Test1 class 之前被字符串池 table 引用时,此行为才有可能。
s1 引用 HEAP 上的字符串对象“Java”,s1.intern() returns 引用字符串池对象(因为“Java”已经被字符串池引用).
这就是为什么 s1 == s2 returns 为假。

但是当加载 Test2 class 时,字符串池 table 未引用“Udayan”。
s3 引用 HEAP 上的字符串对象“Udayan”,s3.intern() 将 s3 引用的字符串对象添加到字符串池和 returns 相同的引用。这意味着 s3 和 s4 指的是同一个对象。
这就是为什么 s3 == s4 returns 为真。

如果我的观察是正确的,那么这意味着 字符串池最初不是空的。
字符串池最初包含“Java”、“java”、“Oracle”等字符串对象。

谁能证实一下?

这取决于你认为的 "initially"。

JVM 启动时,字符串池为空。然而,当各种基本 JDK classes 在 Test1 class 被加载之前被加载和初始化时,其中一些添加 Strings 就不足为奇了字符串池。 "Java" 必须是这些 String 之一。

并且 JLS 中没有任何内容阻止 Java 的开发人员在较新的 JDK 版本中的 classes 的初始化中引入新的 String 文字。因此,您注意到 JDK 7 和 JDK 8 之间的差异并不奇怪。