字符串常量池、intern方法的奥秘

字符串常量池、intern方法

标签(空格分隔): 随手记


本文JDK是java8,如果涉及到其他版本会特殊说明

疑问

程序片段一:

    String s = new String("1");
    s.intern();
    String s2 = "1";
    System.out.println(s == s2);

程序片段二 :

 String s3 = new String("1") + new String("1");
 s3.intern();
 String s4 = "11";
 System.out.println(s3 == s4);

上面两个片段的运行结果如何?

前置

要明白上面程序的运行结果,首先要明白字符串什么时候 加入到常量池?intern方法的作用是什么?

字符串什么时候加入到常量池

这个问题在文章《String常量池、字符串拼接》中有详细的分析,目前得出的结论是<font color="red">当字符串声明的时候会加入常量池</font>

intern方法的作用

来看一看源码的注释

 /**
     * Returns a canonical representation for the string object.
     * <p>
     * A pool of strings, initially empty, is maintained privately by the
     * class {@code String}.
     * <p>
     * When the intern method is invoked, if the pool already contains a
     * string equal to this {@code String} object as determined by
     * the {@link #equals(Object)} method, then the string from the pool is
     * returned. Otherwise, this {@code String} object is added to the
     * pool and a reference to this {@code String} object is returned.
     * <p>
     * It follows that for any two strings {@code s} and {@code t},
     * {@code s.intern() == t.intern()} is {@code true}
     * if and only if {@code s.equals(t)} is {@code true}.
     * <p>
     * All literal strings and string-valued constant expressions are
     * interned. String literals are defined in section 3.10.5 of the
     * <cite>The Java&trade; Language Specification</cite>.
     *
     * @return  a string that has the same contents as this string, but is
     *          guaranteed to be from a pool of unique strings.
     */

来看这一段:
<font color="red">
When the intern method is invoked, if the pool already contains a string equal to this {@code String} object as determined by the {@link #equals(Object)} method, then the string from the pool is returned. Otherwise, this {@code String} object is added to the pool and a reference to this {@code String} object is returned.
</font>

当这个方法被调用的时候,如果在常量池中通过equals方法能找到跟这个字符串相等的字符串,那么返回那个字符串。否则的话,把这个字符串加入到常量池并且返回这个对象的引用。

运行

运行程序片段一:

false

运行程序片段二:

true

分析

程序片段一:

    String s = new String("1");
    s.intern();
    String s2 = "1";
    System.out.println(s == s2);
  • 首先我们看到了一个"1"字符串的声明,所以把"1"会加入到常量池中
  • 然后会在堆上面创建一个"1",并把地址给到s
  • 调用intern方法,去常量池寻找值为"1"的字符串,原来就存在,所以返回常量池的引用,但是引用没有赋值给任何一个变量
  • 又声明了一个"1",但是从常量池找到了"1",所以直接返回常量池"1"的地址
  • 比较堆"1"和常量池"1"的地址,发现是不一样的

程序片段二:

    String s3 = new String("1") + new String("1");
    s3.intern();
    String s4 = "11";
    System.out.println(s3 == s4);
  • 首先我们看到了"1"和"1"的声明,毫无疑问首先会把"1"加入到常量池,第二个"1"的地址跟第一个是一样的
  • 然后拼接出来一个char数组char c=['1','1'],并通过toString的方法在堆上创建了一个"11"字符串,然后把地址给到了s3
  • 调用intern方法,去常量池寻找值为"11"的字符串,发现原来没有,于是准备把"11"加入到常量池,这里是把堆上"11"的地址加进去了
  • 判断堆上的"11"的地址和常量池中"11"的地址,发现是一样的

延伸

程序片段二在jdk6下会有不同的结果

false

为什么是false?

因为jdk6中不会把堆上的引用直接放在字符串常量池里面,而是会重新创建一个属于常量池的字符串,所以比较堆上字符串和常量池字符串的地址的时候就会返回false

ps:这也许跟字符串常量池的位置有关系,在jdk6的时候,字符串常量池是在运行时常量池里面的,是存在于永久代,到了jdk7以后,字符串常量池就跟运行时常量池分开了,在堆上分配空间。可能这就是jdk7能把堆上的地址直接放到字符串常量池的原因吧。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容