今天在看一本書的時候注意到一個String的intern()方法,平常沒用過,只是看過這個方法,也沒去仔細看過這個方法。所以今天看了一下。個人覺得String類別加入這個方法可能是為了提升一點點效能,因為從常數池取資料比從堆裡面去資料快一些。 (個人感覺)
API上的那幾句關於這個方法,其實總結一句就是呼叫這個方法之後把字串物件加入常數池中,常數池我們都知道他是存在於方法區的,他是方法區的一部分,而方法區是線程共享的,所以常數池也就是線程共享的,但是他並不是線程不安全的,他其實是線程安全的,他僅僅是讓有相同值的引用指向同一個位置而已,如果引用值變化了,但是常量池中沒有新的值,那麼就會新開闢一個常數結果來交給新的引用,而並非像線程不同步那樣,針對同一個對象,new出來的字串和直接賦值給變數的字串存放的位置是不一樣的,前者是在堆裡面,而後者在常量池裡面,另外,在做字符串拼接操作,也就是字符串相"+"的時候,得出的結果是存在在常數池或堆裡面,這個是根據情況不同不一定的,我寫了幾行程式碼測試了一下。
先上結果:
1.直接定義字串變數的時候賦值,如果表達式右邊只有字串常數,那麼就是把變數存放在常數池裡面。
2.new出來的字串就是存放在堆裡面。
3.對字串進行拼接操作,也就是做"+"運算的時候,分2中情況:
i.表達式右邊是純字串棧裡面。
ii.表達式右邊如果存在字串引用,也就是字串物件的句柄,那麼就存放在堆裡面。
String str1 = "aaa"; String str2 = "bbb"; String str3 = "aaabbb"; String str4 = str1 + str2; String str5 = "aaa" + "bbb"; System.out. print ln(str3 == str4); // false System.out.println(str3 == str4.intern()); // true System.out.println(str3 == str5);// true
結果:str1、str2、str3、str5都是存在於常數池,str4由於表達式右半邊有引用類型,所以str4存在於堆內存,而str5表達式右邊沒有引用類型,是純字串常數,就存放在了常數池裡面。其實Integer這種包裝類型的-128 ~ +127也是存放在常數池裡面,例如Integer i1 = 10;Integer i2 = 10; i1 == i2結果是true,估計也是為了性能優化。
【相關推薦】
1. Java免費影片教學
#以上是分析Java中的intern()的作用的詳細內容。更多資訊請關注PHP中文網其他相關文章!