深入理解Java new String()方法
String str1 = new String('aa');
答案是兩個(gè)“aa”對(duì)象和String對(duì)象
其中有一項(xiàng)是常量池常量池在Class文件被加載的時(shí)候,會(huì)被加載進(jìn)內(nèi)存中的方法區(qū)中的運(yùn)行時(shí)常量池,而運(yùn)行時(shí)常量池里就包括字符串常量池,Class文件中的字符串在類加載時(shí)就會(huì)加載到字符串常量池中去
不過(guò)在周志明老師在深入java虛擬機(jī)中有說(shuō)到,到了JDK1.7時(shí),字符串常量池就被移出了方法區(qū),轉(zhuǎn)移到了堆里了。
String str1 = new String(“aa”);'aa'就是被加載進(jìn)去的字符串,我們可以看看Class文件
這里的aa在之后類加載的時(shí)候,會(huì)在字符串常量池里創(chuàng)建一個(gè) 'aa'對(duì)象,這是第一個(gè)對(duì)象類加載完成了之后,那就要開(kāi)始正式執(zhí)行代碼了,執(zhí)行該行代碼時(shí)new一個(gè)'aa'的String對(duì)象存放在Java堆中,這是第二個(gè)對(duì)象創(chuàng)建完第二個(gè)對(duì)象后,虛擬機(jī)棧上的str1將會(huì)指向第二個(gè)對(duì)象,也就是堆上的對(duì)象
問(wèn)題二:輸出結(jié)果是true還是false?String str1 = new String('aa'); String str2 = 'aa'; System.out.println(str1 == str2);
答案很明顯是false,因?yàn)閮蓚€(gè)變量指向的地址不同,一個(gè)指向字符串常量池,一個(gè)指向堆上的對(duì)象,而==比較的就是地址。
問(wèn)題三:輸出結(jié)果是true?String str1 = new String('aa'); str1.intern(); String str2 = 'aa'; System.out.println(str1 == str2);
首先我們來(lái)了解一下intern方法intern的處理是 先判斷字符串常量是否在字符串常量池中,如果存在直接返回該常量,如果沒(méi)有找到,說(shuō)明該字符串常量在堆中,則處理是把堆區(qū)該對(duì)象的引用加入到字符串常量池中,以后別人拿到的是該字符串常量的引用,實(shí)際存在堆中。
也就是說(shuō)現(xiàn)在字符串常量池中的'aa'實(shí)際上是指向堆上的String對(duì)象的?所以結(jié)果是true?并不是,結(jié)果還是false
回到問(wèn)題一
String str1 = new String('aa');
這段代碼創(chuàng)建了兩個(gè)對(duì)象,而第一個(gè)就是在字符串常量池中的,而intern方法在判斷時(shí)會(huì)發(fā)現(xiàn)字符串常量池中已經(jīng)存在'aa'對(duì)象了,所以它就不用把字符串常量池中添加一個(gè)指向堆上的String對(duì)象的地址了所以最后intern方法只是返回了'aa'對(duì)象,并沒(méi)有做任何修改
所以還是str1指向堆,str2指向字符串常量池,結(jié)果為false
問(wèn)題四:那要怎么樣才能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”);拼接起來(lái)的方式,因此在編譯后,Class文件中的常量池寫(xiě)入的是'a'對(duì)象而不是'aa'對(duì)象,如下圖:
因此intern方法在判斷時(shí)會(huì)發(fā)現(xiàn)字符串常量池中并沒(méi)有'aa'對(duì)象,于是它就把堆中String對(duì)象的引用加入到字符串常量池中。之后創(chuàng)建str4的時(shí)候,str4就會(huì)先在字符串常量池中先查找有沒(méi)有'aa',于是它找到了intern放入的引用,并把這個(gè)引用賦給str4所以str3和str4都是同一個(gè)引用,str3==str4,為true
問(wèn)題五:那么這段代碼又創(chuàng)建了幾個(gè)對(duì)象?String str3 = new String('a') + new String('a');
答案是五個(gè)
因?yàn)槭褂?號(hào)的String字符串拼接,底層其實(shí)都是先創(chuàng)建一個(gè)StringBuilder對(duì)象,然后調(diào)用append方法把要+的字符串都append進(jìn)去,最后toString創(chuàng)建一個(gè)新的String對(duì)象如下圖:
紅色的地方就是new出來(lái)對(duì)象的語(yǔ)句,而綠色則是兩次append四個(gè)紅色一共四個(gè)對(duì)象,再加上字符串常量池上創(chuàng)建的'a'對(duì)象,一共五個(gè)
這也正是為什么阿里巴巴代碼規(guī)范中不建議在for循環(huán)里使用+號(hào)拼接字符串
String str1 = 'aaa';String str2 = 'bbb';String str4 = str1 + str2;
這個(gè)的String str4 = str1 + str2;創(chuàng)建了兩個(gè)對(duì)象,StringBuilder和toString時(shí)生成的String對(duì)象
那下面這段呢?是'aaa'對(duì)象加'bbb'對(duì)象加StringBuilder和toString時(shí)生成的String對(duì)象一共四個(gè)對(duì)象嗎?
String str5 = 'aaa' + 'bbb';
很可惜這段只創(chuàng)建了1個(gè)對(duì)象java編譯器在編譯這段的時(shí)候做了優(yōu)化,實(shí)際上'aaa'+'bbb'會(huì)先拼接成'aaabbb'之后才開(kāi)始編譯,也就是說(shuō)這段代碼等于是String str5 = “aaabbb”如下圖:(code里面沒(méi)有任何new操作)
到此這篇關(guān)于深入理解Java new String()的文章就介紹到這了,更多相關(guān)Java new String()內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!
相關(guān)文章:
1. Intellij IDEA 2020.3 配置教程詳解2. idea給項(xiàng)目打war包的方法步驟3. IntelliJ IDEA設(shè)置編碼格式的方法4. Python importlib模塊重載使用方法詳解5. 兩行Javascript代碼生成UUID的方法6. XML入門(mén)精解之結(jié)構(gòu)與語(yǔ)法7. IntelliJ IDEA刪除類的方法步驟8. ASP基礎(chǔ)入門(mén)第八篇(ASP內(nèi)建對(duì)象Application和Session)9. 使用 kind 和 Docker 啟動(dòng)本地的 Kubernetes環(huán)境10. 解決python中import文件夾下面py文件報(bào)錯(cuò)問(wèn)題
