深入理解Java new String()方法

問題一:這段代碼創建瞭幾個對象?

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操作)

在這裡插入圖片描述

到此這篇關於深入理解Java new String()的文章就介紹到這瞭,更多相關Java new String()內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: