教妹學(xué) Java之Intern
“哥,你發(fā)給我的那篇文章我看了,結(jié)果直接把我給看得不想學(xué) Java 了!”三妹氣沖沖地說。
“哪一篇啊?”看著三妹面色沉重,我關(guān)心地問到。
“就是美團技術(shù)團隊深入解析 String.intern() 那篇啊!”三妹回答。
https://tech.meituan.com/2014/03/06/in-depth-understanding-string-intern.html
“哦,我想起來了,不挺好一篇文章嘛,深入淺出,精品中的精品,看完后你應(yīng)該對 String 的 intern 徹底理解了才對呀。”
“好是好,但我就是看不懂!”三妹委屈地說,“哥,還是你親自給我講講吧?”
“好吧,上次學(xué)的字符串常量池你都搞清楚了吧?”
“嗯。”三妹微微的點了點頭。
要理解美團技術(shù)團隊的這篇文章,你只需要記住這幾點內(nèi)容:
第一,使用雙引號聲明的字符串對象會保存在字符串常量池中。
第二,使用 new 關(guān)鍵字創(chuàng)建的字符串對象會先從字符串常量池中找,如果沒找到就創(chuàng)建一個,然后再在堆中創(chuàng)建字符串對象;如果找到了,就直接在堆中創(chuàng)建字符串對象。
第三,針對沒有使用雙引號聲明的字符串對象來說,就像下面代碼中的 s1 那樣:
- String s1 = new String("二哥") + new String("三妹");
如果想把 s1 的內(nèi)容也放入字符串常量池的話,可以調(diào)用 intern() 方法來完成。
不過,需要注意的是,Java 7 的時候,字符串常量池從永久代中移動到了堆中,雖然此時永久代還沒有完全被移除。Java 8 的時候,永久代被徹底移除。
這個變化也直接影響了 String.intern() 方法在執(zhí)行時的策略,Java 7 之前,執(zhí)行 String.intern() 方法的時候,不管對象在堆中是否已經(jīng)創(chuàng)建,字符串常量池中仍然會創(chuàng)建一個內(nèi)容完全相同的新對象;Java 7 之后呢,由于字符串常量池放在了堆中,執(zhí)行 String.intern() 方法的時候,如果對象在堆中已經(jīng)創(chuàng)建了,字符串常量池中就不需要再創(chuàng)建新的對象了,而是直接保存堆中對象的引用,也就節(jié)省了一部分的內(nèi)存空間。
“三妹,來猜猜這段代碼輸出的結(jié)果吧。”我說。
- String s1 = new String("二哥三妹");
- String s2 = s1.intern();
- System.out.println(s1 == s2);
“哥,這我完全猜不出啊,還是你直接解釋吧。”三妹說。
“好吧。”
第一行代碼,字符串常量池中會先創(chuàng)建一個“二哥三妹”的對象,然后堆中會再創(chuàng)建一個“二哥三妹”的對象,s1 引用的是堆中的對象。
第二行代碼,對 s1 執(zhí)行 intern() 方法,該方法會從字符串常量池中查找“二哥三妹”這個字符串是否存在,此時是存在的,所以 s2 引用的是字符串常量池中的對象。
也就意味著 s1 和 s2 的引用地址是不同的,一個來自堆,一個來自字符串常量池,所以輸出的結(jié)果為 false。
“來看一下運行結(jié)果。”我說。
- false
“我來畫幅圖,幫助你理解下。”看到三妹驚訝的表情,我耐心地說。
“這下理解了吧?”我問三妹。
“嗯嗯,一下子就豁然開朗了!”三妹說。
“好,我們再來看下面這段代碼。”
- String s1 = new String("二哥") + new String("三妹");
- String s2 = s1.intern();
- System.out.println(s1 == s2);
“難道也輸出 false ?”三妹有點不確定。
“不,這段代碼會輸出 true。”我否定了三妹的猜測。
“為啥呀?”三妹迫切地想要知道答案。
第一行代碼,會在字符串常量池中創(chuàng)建兩個對象,一個是“二哥”,一個是“三妹”,然后在堆中會創(chuàng)建兩個匿名對象“二哥”和“三妹”(可以暫時忽略),最后還有一個“二哥三妹”的對象,s1 引用的是堆中“二哥三妹”這個對象。
第二行代碼,對 s1 執(zhí)行 intern() 方法,該方法會從字符串常量池中查找“二哥三妹”這個對象是否存在,此時不存在的,但堆中已經(jīng)存在了,所以字符串常量池中保存的是堆中這個“二哥三妹”對象的引用,也就是說,s2 和 s1 的引用地址是相同的,所以輸出的結(jié)果為 true。
“來看一下運行結(jié)果。”我胸有成竹地說。
- true
“我再來畫幅圖,幫助你理解下。”
“哇,我明白了!”三妹長舒一口氣,大有感慨 intern 也沒什么難理解的意味。
不過需要注意的是,盡管 intern 可以確保所有具有相同內(nèi)容的字符串共享相同的內(nèi)存空間,但也不要爛用 intern,因為任何的緩存池都是有大小限制的,不能無緣無故就占用了相對稀缺的緩存空間,導(dǎo)致其他字符串沒有坑位可占。
另外,字符串常量池本質(zhì)上是一個固定大小的 StringTable,如果放進去的字符串過多,就會造成嚴(yán)重的哈希沖突,從而導(dǎo)致鏈表變長,鏈表變長也就意味著字符串常量池的性能會大幅下降,因為要一個一個找是需要花費時間的。
“好了,三妹,關(guān)于 String 的 intern 就講到這吧,這次理解了吧?”我問。
“哥,你真棒!”
看到三妹一點一滴的進步,我也感到由衷的開心。
本文轉(zhuǎn)載自微信公眾號「沉默王二」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系沉默王二公眾號。