写技术文章其实是个很好的学习方式。首先你得自己摸清楚原理,然后才能尝试去表达出来。你写出来的东西,别人看了,给予你反馈,也是一个互相学习的过程。这不,前几天碰到一个读者提出的一个问题,让我发现了自己文章中的一个疏漏,下面就来说说这个问题。
在我之前的一篇文章你并不了解 String 中出了这样一道题目:
String str1 = new String("j") + new String("ava"); // 1 str1.intern(); // 2 String str2 = "java"; // 3 System.out.println(str1 == str2); // 4 复制代码
看一遍,想一遍,如果还不能准确给出答案的话可以再仔细看看我的那篇文章。正常情况下,答案应该是 true
。贴一下之前的讲解:
经过编译,j 、ava 和 java 进入 Class 常量池 中。 类加载阶段并不会创建实例,驻留字符串常量池。到运行期,第一行代码中会创建 j 、ava 实例并驻留常量池,+ 会被 JVM 自动优化为 StringBuilder ,拼接出 java 字符串,将 str1 指向该字符串实例。需要注意的是,这里不会将 java 驻留到常量池。第二行代码调用了 intern(),由于此时常量池中没有 java,所以将 str1 的引用存入了常量池。第三行代码,ldc 指令发现常量池中就有 java,直接返回常量池中其对应的引用,并赋给 str2。所以 str1 和 str2 是相等的。
看起来说的还是挺有道理的,我自己放到 IDEA 中执行也的确是 true
。但是有一位读者给我评论说他执行下来是 false
。我的第一反应就是 “我的代码肯定没错,应该是他写错了” ,就加了这兄弟的微信,截图给我看了看代码。然后我就蒙了,果然是 false
。同样的代码,却是不同的执行结果。
那么,究竟 str1
和 str2
为什么会不相等呢?
我们再来分析一下代码,第一行:
String str1 = new String("j") + new String("ava"); // 1 复制代码
从这句话中可以肯定的是, str1
指向堆中的一个 java
字符串实例,且这个字符串是在堆中新创建的。再看第二和第三行:
str1.intern(); // 2 String str2 = "java"; // 3 复制代码
str1.intern()
, str1
是指向堆中的一个 java
字符串实例的,调用 intern() 的话,此时就有两种情况:
java java
第一种情况,就是我上篇文章中的分析, str1
驻留到字符串常量池,结果是 true
。第二种情况,字符串常量池中已经有了 java
,此时再执行 str1.intern()
就会直接返回字符串常量池中 java
字符串对应的引用,并不会将 str1
驻留到字符串常量池。 String str2 = "java";
一执行, str2
等于字符串常量池中的 java
对应的引用。而 str1
是新建在堆中的 java
字符串的引用,自然而然,比较结果是 false
。
这么分析下来,这位读者无疑是第二种情况了。但是为什么同样在 main()
方法中直接执行这几行代码,结果会不一样呢?为什么 main()
方法中的代码还没有执行,字符串常量池中就已经有 java
字符串的引用了呢?
对啊,为什么会这样呢?等等, java
,这个字符串是不是有点特殊。我尝试着让这位读者换一个特殊点的字符串再运行一次,竟然还让我蒙对了,这下打印 true
了。看到这里,你应该明白了,在 JVM
启动的过程中,字符串常量池已经在发挥作用了,在 main()
方法运行之前,一些字符串引用已经驻留在字符串常量池,比如上面的 java
,但也不是百分之百的,我手里的 Ubuntu 18.04
就一直打印的是 true
,你们也可以掏出电脑来试一试。
对于 String.intern()
方法,我们只需要搞清楚当前字符串常量池是否已经驻留该字符串引用,已驻留和未驻留将导致不同的执行逻辑。
这么一来,我和这位读者都彻彻底底的搞清楚了 intern()
方法,下次再遇到类似的面试题应该都不是问题了。所以呢,也欢迎大家多多提出自己的评论和想法,可以在掘金文章评论,可以微信搜索 秉心说 或者扫描文末二维码关注公众号私信我,也可以直接加我微信 bingxinshuo_ ,微信后台回复 秉心说 也可以加我微信。力所能及的问题我都会尽量解答。
文章越写越短了,走进 JDK 系列好久不出新文章了,主要是最近输出大于输入,白天要上班,晚上还要奶孩子,每次文章大概都是十二点左右发出来的。不过好在最近的文章阅读量都还不错,也才有了写下去的动力。
大家也可以关注我的公众号 "秉心说",这个名字的由来,其实很简单,就是我女儿的名字。"秉心识本源,于事少凝滞",也是我的公众号自动回复的一句话,既是对孩子的期许,也是对自己的期许,也希望大家都可以做到 秉持内心,保持本源
。(大白话翻译一下,就是别忘了自己是个程序猿!)
说了一些题外话,后续还是会继续保持输入和输出,专注 Java/Android 原创知识分享,不妨点个赞再走吧!
文章首发于微信公众号: 秉心说
, 专注 Java 、 Android 原创知识分享,LeetCode 题解,欢迎关注!