問題一:這段代碼創建了幾個物件?
String str1 = new String("aa");
答案是兩個
“aa”物件和String物件
其中有一項是常量池
常量池在Class檔案被加載的時候,會被加載進記憶體中的方法區中的運行時常量池,而運行時常量池里就包括字串常量池,Class檔案中的字串在類加載時就會加載到字串常量池中去

不過在周志明老師在深入java虛擬機中有說到,到了JDK1.7時,字串常量池就被移出了方法區,轉移到了堆里了,
String str1 = new String(“aa”);
"aa"就是被加載進去的字串,我們可以看看Class檔案

- 這里的aa在之后類加載的時候,會在字串常量池里創建一個 "aa"物件,這是第一個物件
- 類加載完成了之后,那就要開始正式執行代碼了,執行該行代碼時new一個"aa"的String物件存放在Java堆中,這是第二個物件
- 創建完第二個物件后,虛擬機堆疊上的str1將會指向第二個物件,也就是堆上的物件
問題二:輸出結果是true還是false?
String str1 = new String("aa");
String str2 = "aa";
System.out.println(str1 == str2);
答案很明顯是false,因為兩個變數指向的地址不同,一個指向字串常量池,一個指向堆上的物件,而==比較的就是地址,
問題三:輸出結果是true?
String str1 = new String("aa");
str1.intern();
String str2 = "aa";
System.out.println(str1 == str2);
首先我們來了解一下intern方法
intern的處理是 先判斷字串常量是否在字串常量池中,如果存在直接回傳該常量,如果沒有找到,說明該字串常量在堆中,則處理是把堆區該物件的參考加入到字串常量池中,以后別人拿到的是該字串常量的參考,實際存在堆中,
也就是說現在字串常量池中的"aa"實際上是指向堆上的String物件的?所以結果是true?
并不是,結果還是false
回到問題一
String str1 = new String("aa");
這段代碼創建了兩個物件,而第一個就是在字串常量池中的,而intern方法在判斷時會發現字串常量池中已經存在"aa"物件了,所以它就不用把字串常量池中添加一個指向堆上的String物件的地址了
所以最后intern方法只是回傳了"aa"物件,并沒有做任何修改
所以還是str1指向堆,str2指向字串常量池,結果為false
問題四:那要怎么樣才能true?
String str3 = new String("a") + new String("a");
str3.intern();
String str4 = "aa";
System.out.println(str3 == str4);
這里列印的結果就是true了
這里的str3生成的方式不再是new String(“aa”);
而是new String(“a”) + new String(“a”);拼接起來的方式,因此在編譯后,Class檔案中的常量池寫入的是"a"物件而不是"aa"物件,如下圖:

因此intern方法在判斷時會發現字串常量池中并沒有"aa"物件,于是它就把堆中String物件的參考加入到字串常量池中,
之后創建str4的時候,str4就會先在字串常量池中先查找有沒有"aa",于是它找到了intern放入的參考,并把這個參考賦給str4
所以str3和str4都是同一個參考,str3==str4,為true
問題五:那么這段代碼又創建了幾個物件?
String str3 = new String("a") + new String("a");
答案是五個
因為使用+號的String字串拼接,底層其實都是先創建一個StringBuilder物件,然后呼叫append方法把要+的字串都append進去,最后toString創建一個新的String物件如下圖:

紅色的地方就是new出來物件的陳述句,而綠色則是兩次append
四個紅色一共四個物件,再加上字串常量池上創建的"a"物件,一共五個
這也正是為什么阿里巴巴代碼規范中不建議在for回圈里使用+號拼接字串
String str1 = "aaa";
String str2 = "bbb";
String str4 = str1 + str2;
這個的String str4 = str1 + str2;創建了兩個物件,StringBuilder和toString時生成的String物件
那下面這段呢?是"aaa"物件加"bbb"物件加StringBuilder和toString時生成的String物件一共四個物件嗎?
String str5 = "aaa" + "bbb";
很可惜這段只創建了1個物件
java編譯器在編譯這段的時候做了優化,實際上"aaa"+"bbb"會先拼接成"aaabbb"之后才開始編譯,也就是說這段代碼等于是String str5 = “aaabbb”
如下圖:(code里面沒有任何new操作)

轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/289108.html
標籤:其他
上一篇:JDK JRE JVM的區別
