溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java new String()方法有哪些作用

發(fā)布時(shí)間:2021-06-18 09:09:29 來源:億速云 閱讀:1337 作者:chen 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“Java new String()方法有哪些作用”的有關(guān)知識(shí),在實(shí)際案例的操作過程中,不少人都會(huì)遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

問題一:這段代碼創(chuàng)建了幾個(gè)對(duì)象?

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ì)加載到字符串常量池中去

Java new String()方法有哪些作用

不過在周志明老師在深入java虛擬機(jī)中有說到,到了JDK1.7時(shí),字符串常量池就被移出了方法區(qū),轉(zhuǎn)移到了堆里了。

String str1 = new String(“aa”);
"aa"就是被加載進(jìn)去的字符串,我們可以看看Class文件

Java new String()方法有哪些作用

這里的aa在之后類加載的時(shí)候,會(huì)在字符串常量池里創(chuàng)建一個(gè) "aa"對(duì)象,這是第一個(gè)對(duì)象類加載完成了之后,那就要開始正式執(zhí)行代碼了,執(zhí)行該行代碼時(shí)new一個(gè)"aa"的String對(duì)象存放在Java堆中,這是第二個(gè)對(duì)象創(chuàng)建完第二個(gè)對(duì)象后,虛擬機(jī)棧上的str1將會(huì)指向第二個(gè)對(duì)象,也就是堆上的對(duì)象

問題二:輸出結(jié)果是true還是false?

String str1 = new String("aa");
 String str2 = "aa";
 System.out.println(str1 == str2);

答案很明顯是false,因?yàn)閮蓚€(gè)變量指向的地址不同,一個(gè)指向字符串常量池,一個(gè)指向堆上的對(duì)象,而==比較的就是地址。

問題三:輸出結(jié)果是true?

String str1 = new String("aa");
 str1.intern();
 String str2 = "aa";
 System.out.println(str1 == str2);

首先我們來了解一下intern方法
intern的處理是 先判斷字符串常量是否在字符串常量池中,如果存在直接返回該常量,如果沒有找到,說明該字符串常量在堆中,則處理是把堆區(qū)該對(duì)象的引用加入到字符串常量池中,以后別人拿到的是該字符串常量的引用,實(shí)際存在堆中。

也就是說現(xiàn)在字符串常量池中的"aa"實(shí)際上是指向堆上的String對(duì)象的?所以結(jié)果是true?
并不是,結(jié)果還是false

回到問題一

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ì)象,并沒有做任何修改

所以還是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"對(duì)象而不是"aa"對(duì)象,如下圖:

Java new String()方法有哪些作用

因此intern方法在判斷時(shí)會(huì)發(fā)現(xiàn)字符串常量池中并沒有"aa"對(duì)象,于是它就把堆中String對(duì)象的引用加入到字符串常量池中。
之后創(chuàng)建str4的時(shí)候,str4就會(huì)先在字符串常量池中先查找有沒有"aa",于是它找到了intern放入的引用,并把這個(gè)引用賦給str4
所以str3和str4都是同一個(gè)引用,str3==str4,為true

問題五:那么這段代碼又創(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ì)象如下圖:

Java new String()方法有哪些作用

紅色的地方就是new出來對(duì)象的語句,而綠色則是兩次append
四個(gè)紅色一共四個(gè)對(duì)象,再加上字符串常量池上創(chuàng)建的"a"對(duì)象,一共五個(gè)

這也正是為什么阿里巴巴代碼規(guī)范中不建議在for循環(huán)里使用+號(hào)拼接字符串

Java new String()方法有哪些作用

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"之后才開始編譯,也就是說這段代碼等于是String str5 = “aaabbb”
如下圖:(code里面沒有任何new操作)

Java new String()方法有哪些作用

“Java new String()方法有哪些作用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI