《你并不了解 String》 勘误

写技术文章其实是个很好的学习方式。首先你得自己摸清楚原理,然后才能尝试去表达出来。你写出来的东西,别人看了,给予你反馈,也是一个互相学习的过程。这不,前几天碰到一个读者提出的一个问题,让我发现了自己文章中的一个疏漏,下面就来说说这个问题。

在我之前的一篇文章 你并不了解 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。同样的代码,却是不同的执行结果。

那么,究竟 str1str2 为什么会不相等呢?

我们再来分析一下代码,第一行:

String str1 = new String("j") + new String("ava"); // 1
复制代码

从这句话中可以肯定的是,str1 指向堆中的一个 java 字符串实例,且这个字符串是在堆中新创建的。再看第二和第三行:

str1.intern(); // 2
String str2 = "java"; // 3
复制代码

str1.intern()str1 是指向堆中的一个 java 字符串实例的,调用 intern() 的话,此时就有两种情况:

  1. 字符串常量池中没有 java 字符串的引用
  2. 字符串常量池中已经有了 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 题解,欢迎关注!

猜你喜欢

转载自juejin.im/post/5cb5e3d751882532bb2d1ff7