深入理解Java new String()方法
問題一:這段代碼創(chuàng)建了幾個對象?
String str1 = new String("aa");
答案是兩個
“aa”對象和String對象
其中有一項是常量池
常量池在Class文件被加載的時候,會被加載進(jìn)內(nèi)存中的方法區(qū)中的運行時常量池,而運行時常量池里就包括字符串常量池,Class文件中的字符串在類加載時就會加載到字符串常量池中去
不過在周志明老師在深入java虛擬機(jī)中有說到,到了JDK1.7時,字符串常量池就被移出了方法區(qū),轉(zhuǎn)移到了堆里了。
String str1 = new String(“aa”);
"aa"就是被加載進(jìn)去的字符串,我們可以看看Class文件
這里的aa在之后類加載的時候,會在字符串常量池里創(chuàng)建一個 "aa"對象,這是第一個對象類加載完成了之后,那就要開始正式執(zhí)行代碼了,執(zhí)行該行代碼時new一個"aa"的String對象存放在Java堆中,這是第二個對象創(chuàng)建完第二個對象后,虛擬機(jī)棧上的str1將會指向第二個對象,也就是堆上的對象
問題二:輸出結(jié)果是true還是false?
String str1 = new String("aa"); String str2 = "aa"; System.out.println(str1 == str2);
答案很明顯是false,因為兩個變量指向的地址不同,一個指向字符串常量池,一個指向堆上的對象,而==比較的就是地址。
問題三:輸出結(jié)果是true?
String str1 = new String("aa"); str1.intern(); String str2 = "aa"; System.out.println(str1 == str2);
首先我們來了解一下intern方法
intern的處理是 先判斷字符串常量是否在字符串常量池中,如果存在直接返回該常量,如果沒有找到,說明該字符串常量在堆中,則處理是把堆區(qū)該對象的引用加入到字符串常量池中,以后別人拿到的是該字符串常量的引用,實際存在堆中。
也就是說現(xiàn)在字符串常量池中的"aa"實際上是指向堆上的String對象的?所以結(jié)果是true?
并不是,結(jié)果還是false
回到問題一
String str1 = new String("aa");
這段代碼創(chuàng)建了兩個對象,而第一個就是在字符串常量池中的,而intern方法在判斷時會發(fā)現(xiàn)字符串常量池中已經(jīng)存在"aa"對象了,所以它就不用把字符串常量池中添加一個指向堆上的String對象的地址了
所以最后intern方法只是返回了"aa"對象,并沒有做任何修改
所以還是str1指向堆,str2指向字符串常量池,結(jié)果為false
問題四:那要怎么樣才能true?
String str3 = new String("a") + new String("a"); str3.intern(); String str4 = "aa"; System.out.println(str3 == str4);
這里打印的結(jié)果就是true了
這里的str3生成的方式不再是new String(“aa”);
而是new String(“a”) + new String(“a”);拼接起來的方式,因此在編譯后,Class文件中的常量池寫入的是"a"對象而不是"aa"對象,如下圖:
因此intern方法在判斷時會發(fā)現(xiàn)字符串常量池中并沒有"aa"對象,于是它就把堆中String對象的引用加入到字符串常量池中。
之后創(chuàng)建str4的時候,str4就會先在字符串常量池中先查找有沒有"aa",于是它找到了intern放入的引用,并把這個引用賦給str4
所以str3和str4都是同一個引用,str3==str4,為true
問題五:那么這段代碼又創(chuàng)建了幾個對象?
String str3 = new String("a") + new String("a");
答案是五個
因為使用+號的String字符串拼接,底層其實都是先創(chuàng)建一個StringBuilder對象,然后調(diào)用append方法把要+的字符串都append進(jìn)去,最后toString創(chuàng)建一個新的String對象如下圖:
紅色的地方就是new出來對象的語句,而綠色則是兩次append
四個紅色一共四個對象,再加上字符串常量池上創(chuàng)建的"a"對象,一共五個
這也正是為什么阿里巴巴代碼規(guī)范中不建議在for循環(huán)里使用+號拼接字符串
String str1 = "aaa"; String str2 = "bbb"; String str4 = str1 + str2;
這個的String str4 = str1 + str2;創(chuàng)建了兩個對象,StringBuilder和toString時生成的String對象
那下面這段呢?是"aaa"對象加"bbb"對象加StringBuilder和toString時生成的String對象一共四個對象嗎?
String str5 = "aaa" + "bbb";
很可惜這段只創(chuàng)建了1個對象
java編譯器在編譯這段的時候做了優(yōu)化,實際上"aaa"+"bbb"會先拼接成"aaabbb"之后才開始編譯,也就是說這段代碼等于是String str5 = “aaabbb”
如下圖:(code里面沒有任何new操作)
到此這篇關(guān)于深入理解Java new String()的文章就介紹到這了,更多相關(guān)Java new String()內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
@RequestParam使用defaultValue屬性設(shè)置默認(rèn)值的操作
這篇文章主要介紹了@RequestParam使用defaultValue屬性設(shè)置默認(rèn)值的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02詳解Java實現(xiàn)多種方式的http數(shù)據(jù)抓取
本篇文章主要介紹了Java實現(xiàn)多種方式的http數(shù)據(jù)抓取,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧。2016-12-12Java中BigDecimal的equals方法和compareTo方法的區(qū)別詳析
這篇文章主要給大家介紹了關(guān)于Java中BigDecimal的equals方法和compareTo方法區(qū)別的相關(guān)資料,對于BigDecimal的大小比較,用equals方法的話會不僅會比較值的大小,還會比較兩個對象的精確度,而compareTo方法則不會比較精確度,只比較數(shù)值的大小,需要的朋友可以參考下2023-11-11深入分析JAVA 多線程--interrupt()和線程終止方式
這篇文章主要介紹了JAVA 多線程--interrupt()和線程終止方式的的相關(guān)資料,文中代碼非常細(xì)致,幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-06-06使用SpringBoot的CommandLineRunner遇到的坑及解決
這篇文章主要介紹了使用SpringBoot的CommandLineRunner遇到的坑及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-02-02詳解SpringCloud LoadBalancer 新一代負(fù)載均衡器
這篇文章主要為大家介紹了SpringCloud LoadBalancer新一代負(fù)載均衡器詳解使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01Java模擬rank/over函數(shù)實現(xiàn)獲取分組排名的方法詳解
這篇文章主要為大家詳細(xì)介紹了Java模擬rank()、over()函數(shù)獲取分組排名的方法設(shè)計及實現(xiàn),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-04-04