最近发现一个挺有意思的事情,在jdk1.8和jdk11中分别运行如下代码,获得的结果是不一样的,接下来我们来探究一下他俩的运行结果为什么不一样?
String s3 = new String("1") + new String("1");
s3.intern();
String s4 = "11";
System.out.println(s3 == s4);
想要搞清楚这个问题,首先我们要了解一下intern()这个方法,
intern()的作用
1.如果字符串池中已经存在一个等于该字符串的对象,intern()方法会返回这个已存在的对象的引用。
2.如果字符串池中没有等于该字符串的对象,intern()方法会将该字符串添加到字符串池中,并返回对新添加的字符串对象的引用。
以我们现在对这个方法的理解,是不是应该认为不管什么版本的jdk运行结果都应该为true?,那么我们可以再举一个例子,运行如下代码
public static void main(String[] args) {
String s1 = new String("a");
s1.intern();
String s2 = "a";
System.out.println(s1 == s2); //false
String s3 = new String("a") + new String("a");
s3.intern();
String s4 = "aa";
System.out.println(s3 == s4); //true
}
此刻就会发现两段代码的运行结果不一样,通过查询资料了解到jdk加载字符串常量是在第一次被调用的时候,进行解析并在字符串池中创建对应的String实例
我们可以给以上代码区分一下步骤
a步骤
第一步:此时"a"已经被创建出来,并且s1变量的地址是指向它
第二步:对 s1执行 intern,但是因为"a"这个字符串已经在字符串池中,所以会直接返回原来的引用,但是并没有赋值给任何一个变量。
第三步:s2指向常量池中的"a”
第四步:所以,s1和 s2并不相等
aa步骤
第一步:new 一个 String 对象,并让 s3 指向他。
第二步:对 s3 执行 intern,但是目前字符串池中还没有"aa"这个字符串,于是会把<s3指向的String对象的引用>放入<字符串常量池>
第三步:因为"aa"这个字符串已经在字符串池中,所以会直接返回原来的引用,并赋值给 s4;
第四步:所以,s3和 s4 相等!
为何jdk版本不同会影响执行结果?
经过对上述问题的探究,有一些人可能已经猜出了为什么
//jdk1.8
String s1 = new String("1") + new String("1");
s1.intern();
String s2 = "11";
System.out.println(s1 == s2); //true
//jdk11
String s1 = new String("1") + new String("1");
s1.intern();
String s2 = "11";
System.out.println(s1 == s2); //false
根据上面的经验我们理解的应该是s1=s2的结果应该为true才对,那么为什么jdk11它的结果会是false呢?此时我们不难推断出jdk11字符串常量池中已经存在了"11"这个字符串,感兴趣可以看一下下面这个类,然后多尝试几次是不是和我得出的结论一样JDK/jdk-11.0.2/src/jdk.compiler/com/sun/tools/javac/code/Source.java at 19a6c71e52f3ecd74e4a66be5d0d552ce7175531 · zxiaofan/JDK · GitHub