String到底相不相等?String初始化及String.intern()方法淺析

引言

在各種面試題中經(jīng)常見到類似下述的面試題

寫出main方法的打印結(jié)果

    class test{        
        public static void main(String[] args){
            String s1 = new String("1") + new String("1") + new String("1");
            String s2 = new String("11") + new String("1");
            System.out.println(s1 == s2);
            s2.intern();
            String s3 = "111";
            System.out.println(s2 == s3);
            String s4 = s1.intern();
            System.out.println(s4 == s1);
    
            String s6 = new String("1");
            String s7 = "1";
            String s8 = s6.intern();
            System.out.println(s7 == s6);
            System.out.println(s8 == s6);
            System.out.println(s7 == s8);
        }
    }

答案為:

false
true
false
false
false
true
false
false
true

此題要求對String類的引用有相關(guān)了解藏姐,提煉相關(guān)的要點(diǎn)如下

要點(diǎn)列表

  1. 兩個(gè)對象obj1和obj2望浩,當(dāng)且僅當(dāng)obj1與obj2指向相同的引用(內(nèi)存地址伤锚,對象地址)時(shí)酣栈,ojb1==obj2返回true险胰,其他返回false
  2. String初始化時(shí),若采用String s = "1";這種字面量直接賦值的形式矿筝,則過程為:
    判斷字符串常量池中是否有值與字面量相同的引用起便,有的話s指向這個(gè)引用指向的對象,否則在堆上新建值為字面量的String對象窖维,s指向新建的對象,并在字符串常量池中存儲(chǔ)對應(yīng)的引用
  3. String初始化時(shí)榆综,若采用String s = new String("1");這種new對象的形式,或者使用其他帶參構(gòu)造器陈辱,則過程為:
    在堆上新建字符串實(shí)例奖年,值為相關(guān)值,s直接指向堆上新增的這個(gè)對象
  4. String s = new String("1") + new String("1")等同于String s = new StringBuilder().append("1").append("1").toString();由于StringBuilder的toString方法也是調(diào)用的String的帶參構(gòu)造方法沛贪,
    因此在引用處理時(shí),結(jié)果與3相同
  5. String類的intern()方法執(zhí)行過程為:檢查字符串常量區(qū)是否有相同值(hash)的引用,有的話返回此引用利赋,否則將調(diào)用者放入字符串常量區(qū)水评,并返回調(diào)用者的引用

實(shí)際代碼分析

如下為上述代碼及每一行的分析

    String s1 = new String("1") + new String("1") + new String("1");
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        上面這段代碼,在運(yùn)行時(shí)會(huì)在堆上創(chuàng)建對象"1"并在字符串常量池中創(chuàng)建對應(yīng)的引用媚送,
        同時(shí)由于字符串常量池中沒有值為"111"的對象中燥,因此會(huì)在堆上創(chuàng)建一個(gè)值為"111"的對象,
        s1指向剛創(chuàng)建的這個(gè)"111"對象塘偎,但不會(huì)在常量池中新創(chuàng)建值為"111"的引用疗涉,因此常量池中仍舊
        沒有值為"111"的對象

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        ------------------------------------------------------------------------------------
        String s2 = new String("11") + new String("1");
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        上面這段代碼,在運(yùn)行時(shí)會(huì)在堆上創(chuàng)建對象"11"并在字符串常量池中創(chuàng)建對應(yīng)的引用吟秩,
        同時(shí)由于字符串常量池中仍舊沒有值為"111"的對象咱扣,因此會(huì)在堆上創(chuàng)建一個(gè)值為"111"的對象,
        s2指向剛創(chuàng)建的這個(gè)"111"對象涵防,
        注意闹伪,此"111"對象地址與s1不同,此時(shí)在堆上存在兩個(gè)值都為"111"的String對象

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        addr3("11")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        s2 -> addr4
        ------------------------------------------------------------------------------------
        System.out.println(s1 == s2);
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        根據(jù)上面的運(yùn)行過程可知壮池,s1和s2分別指向堆上兩個(gè)對象偏瓤,只是堆上的對象恰巧值均為"111",
        因此打印false
     */
    s2.intern();
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        根據(jù)JDK7以后的實(shí)現(xiàn),上面這句代碼會(huì)檢查字符串常量區(qū)椰憋,此時(shí)字符串常量區(qū)中沒有值為"111"
        的引用厅克,因此,會(huì)將s2的引用復(fù)制到字符串常量區(qū)

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        s2 -> addr4
        ------------------------------------------------------------------------------------
     */
        String s3 = "111";
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        執(zhí)行了上述intern()過程后橙依,字符串常量區(qū)中已有值為"111"的引用已骇,根據(jù)實(shí)現(xiàn),s3賦值為常量
        區(qū)中值為"111"對應(yīng)的引用也就是s2指向的堆上那個(gè)對象的地址

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        s2 -> addr4
        s3 -> addr4
        ------------------------------------------------------------------------------------
     */
        System.out.println(s2 == s3);
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        s3和s2指向相同的堆上的對象票编,因此結(jié)果為true
     */
        String s4 = s1.intern();
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        因?yàn)樽址A繀^(qū)中已有值為"111"的引用褪储,因此此方法不會(huì)對字符串常量區(qū)中的值產(chǎn)生影響,
        但因?yàn)閕nter()方法返回的是字符串常量區(qū)中的引用慧域,因此s4指向s2對應(yīng)的對象

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        s2 -> addr4
        s3 -> addr4
        s4 -> addr4
        ------------------------------------------------------------------------------------
     */
        System.out.println(s4 == s1);
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        s4和s1指向堆上的不同的對象鲤竹,因此結(jié)果為false
     */
        String s6 = new String("1");
        String s7 = "1";
        String s8 = s6.intern();
    /*
                                    ↓↓↓↓↓↓↓↓解析↓↓↓↓↓↓↓↓↓
        根據(jù)上面的內(nèi)存狀態(tài)可知,s6仍舊指向堆上一個(gè)新建的值為"1"的對象昔榴,
        s6.inter()返回的是字符串常量區(qū)已有的值為"1"的對象地址辛藻,
        s8也等于這個(gè)地址

                                   ↓↓↓↓↓↓↓↓內(nèi)存狀態(tài)↓↓↓↓↓↓↓↓↓
        --------------------------------------堆--------------------------------------------
        addr1("1")
        addr2("111")
        addr3("11")
        addr4("111")
        addr5("1")
        ------------------------------------------------------------------------------------

        ---------------------------------字符串常量池---------------------------------------
        addr1("1")
        addr3("11")
        addr4("111")
        ------------------------------------------------------------------------------------

        ---------------------------------------變量-----------------------------------------
        s1 -> addr2
        s2 -> addr4
        s3 -> addr4
        s4 -> addr4
        s6 -> addr5
        s7 -> addr1
        s8 -> addr1
        ------------------------------------------------------------------------------------


     */
        System.out.println(s7 == s6);
        System.out.println(s8 == s6);
        System.out.println(s7 == s8);
    /*
        根據(jù)上面的內(nèi)存狀態(tài)可知,打印結(jié)果為:
        false
        false
        true
     */

延申

  1. 由于使用有參構(gòu)造器來初始化變量時(shí)互订,總會(huì)在堆上新建變量吱肌,因此在極限情況下,確有可能在初始化階段造成OOM仰禽,解決方法是盡量使用字面量初始化字符串
  2. String內(nèi)部字符串常量區(qū)的實(shí)現(xiàn)方式不同氮墨,openjdk中纺蛆,實(shí)現(xiàn)類似HashMap,當(dāng)放入過多常量時(shí)规揪,插入與查找也會(huì)產(chǎn)生部分性能損耗桥氏,因此,調(diào)用String的intern()方法也要看情況確定
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末猛铅,一起剝皮案震驚了整個(gè)濱河市字支,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌奸忽,老刑警劉巖堕伪,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異栗菜,居然都是意外死亡欠雌,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進(jìn)店門苛萎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來桨昙,“玉大人,你說我怎么就攤上這事腌歉⊥芾遥” “怎么了?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵翘盖,是天一觀的道長桂塞。 經(jīng)常有香客問我,道長馍驯,這世上最難降的妖魔是什么阁危? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮汰瘫,結(jié)果婚禮上狂打,老公的妹妹穿的比我還像新娘。我一直安慰自己混弥,他們只是感情好趴乡,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蝗拿,像睡著了一般晾捏。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上哀托,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天惦辛,我揣著相機(jī)與錄音,去河邊找鬼仓手。 笑死胖齐,一個(gè)胖子當(dāng)著我的面吹牛玻淑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播市怎,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼岁忘,長吁一口氣:“原來是場噩夢啊……” “哼辛慰!你這毒婦竟也來了区匠?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤帅腌,失蹤者是張志新(化名)和其女友劉穎驰弄,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體速客,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡戚篙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了溺职。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片岔擂。...
    茶點(diǎn)故事閱讀 39,690評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖浪耘,靈堂內(nèi)的尸體忽然破棺而出乱灵,到底是詐尸還是另有隱情,我是刑警寧澤七冲,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布痛倚,位于F島的核電站,受9級特大地震影響澜躺,放射性物質(zhì)發(fā)生泄漏蝉稳。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一掘鄙、第九天 我趴在偏房一處隱蔽的房頂上張望耘戚。 院中可真熱鬧,春花似錦操漠、人聲如沸收津。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽朋截。三九已至,卻和暖如春吧黄,著一層夾襖步出監(jiān)牢的瞬間部服,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工拗慨, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留廓八,地道東北人奉芦。 一個(gè)月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像剧蹂,于是被迫代替她去往敵國和親声功。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評論 2 353