1、问题引入
在介绍今天的 知识点之前,先看下面的一段程序:
public static void main(String[] args) {
String s1 = "123456";
String s2 = new String("123") + new String("456");
s2.intern();
System.out.println("result1 = " + (s1 == s2));
System.out.println("result2 = " + (s1 == s2.intern()));
String s3 = new String("123") + new String("123");
s3.intern();
String s4 = "123123";
System.out.println("result3 = " + (s3 == s4));
}
在不运行上述用例前,请回答出result1、result2、result3的输出结果?然后运行示例程序,验证是否和预期的结果一致?想一想,这是为什么呢?
2. 思考
要注意到,在上述程序中,s2和s3的定义类似,不同的是,s1定义在s2之前,而s4定义在s3之后【一定要注意这里的顺序问题】。运行程序后,我们能很快地得出以下结果:
但是,有以下几个为什么需要大家思考:
(1)为什么result3是 true,而 result1 却等于 false?
(2)为什么result1 是 false, 而result2 却等于 true?
(3)intern() 方法究竟干了什么?
3. 解答
首先,我们先看一下intern()方法的说明,如下:
另外,在JVM虚拟机一书中讲到,
结合上面提到的知识点,我们再来分析上述示例中提出的几个为什么。
(1)为什么result3是 true,而 result1 却等于 false?
① String s3 = new String("123") + new String("123"); 执行该段程序后,会在堆中生成一个值为"123123"的字符串【注意:通过new String生成的字符串在堆上,而直接赋值的字符串在池中】;
② s3.intern(); 执行该段程序时,由于池中没有"123123"的字符串常量,根据上图横线部分的介绍,则会在常量池中记录下字符串"123123"的引用,该引用指向s3的变量。
③ String s4 = "123123"; 执行该段程序时,由于池中已有"123123"的引用,因此,会将常量池中记录的"123123"的引用返回给s4。
因此,以上三步解答了为什么result3是true的问题,因为s3和s4指向的都是堆中分配的123123。那为什么result1是false呢?因为程序先执行了 String s1 = "123456"; 此时会在常量池中创建一个字符串常量,而执行String s2 = new String("123") + new String("456");时,在堆中创建一个字符串。再s2.intern();时,不会改变s1和s2的引用,s1和s2仍然指向不同的对象,因此,result1的结果是false。
(2)为什么result1 是 false, 而result2 却等于 true?
result2是s1与s2.intern()的结果比较,而根据intern()方法的说明,由于常量池中有"123456"这个变量,所以s2.intern() 返回的是常量池中s1的引用,因此,result2等于true。
(3)intern() 方法究竟干了什么?
其实,看完前面的介绍,这个问题也就不难回答了。intern()方法主要干了两件事:
1. 常量池中不存在对应的字符串时,调用intern()方法会在常量池中创建该字符串的引用,并返回该字符串的引用;
2. 常量池中存在对应的字符串时,调用intern() 方法会返回该字符串的引用。
最后,附上一篇博文,感兴趣的同学可以看看,https://blog.csdn.net/Mypromise_TFS/article/details/81504137