這篇文章主要介紹“Java字符串常量池和字面量賦值的簡單介紹”,在日常操作中,相信很多人在Java字符串常量池和字面量賦值的簡單介紹問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java字符串常量池和字面量賦值的簡單介紹”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
創(chuàng)新互聯(lián)公司長期為1000多家客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對不同對象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為丹鳳企業(yè)提供專業(yè)的成都網(wǎng)站設(shè)計、網(wǎng)站建設(shè),丹鳳網(wǎng)站改版等技術(shù)服務(wù)。擁有十余年豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開發(fā)。
之所以存在字符串常量池,主要是考慮到String字符串類型比八大基本類型更加復(fù)雜,并且使用的頻率也更加頻繁,經(jīng)常創(chuàng)建字符串對象會造成性能瓶頸,所以采用相似度機(jī)制,將字符串進(jìn)行復(fù)用(享元模式)。
在JDK 1.7 之后(包括1.7),字符串常量池已經(jīng)從方法區(qū)移到了堆中。
String s1 = "古時的風(fēng)箏";
上面是字符串變量的最常用的方式,這種方式叫做字面量聲明,也就用把字符串用雙引號引起來,然后賦值給一個變量。這種情況下會直接將字符串放到字符串常量池中,然后返回給變量。
那這是我再聲明一個內(nèi)容相同的字符串,會發(fā)現(xiàn)字符串常量池中已經(jīng)存在了,那直接指向常量池中的地址即可。
例如上圖所示,聲明了 s1 和 s2,到最后都是指向同一個常量池的地址,所以 s1== s2 的結(jié)果是 true。
用 new String() 的方式,但是基本上不建議這么用,除非有特殊的邏輯需要。
String a = "古時的"; String s2 = new String(a + "風(fēng)箏");
使用這種方式聲明字符串變量的時候,會有兩種情況發(fā)生,
比如在使用 new 之前,已經(jīng)用字面量聲明的方式聲明了一個變量,此時字符串常量池中已經(jīng)存在了相同內(nèi)容的字符串常量。
首先會在堆中創(chuàng)建一個 s2 變量的對象引用;
然后將這個對象引用指向字符串常量池中的已經(jīng)存在的常量;
之前沒有任何地方用到了這個字符串,第一次聲明這個字符串就用的是 new String() 的方式,這種情況下會直接在堆中創(chuàng)建一個字符串對象然后返回給變量。
我看到好多地方說,如果字符串常量池中不存在的話,就先把字符串先放進(jìn)去,然后再引用字符串常量池的這個常量對象,這種說法是有問題的,只是 new String() 的話,如果池中沒有也不會放一份進(jìn)去。
基于 new String() 的這種特性,我們可以得出一個結(jié)論:
String s1 = "古時的風(fēng)箏"; String a = "古時的"; String s2 = new String(a + "風(fēng)箏"); String s3 = new String(a + "風(fēng)箏"); System.out.println(s1==s2); // false System.out.println(s2==s3); // false
以上代碼,肯定輸出的都是 false,因?yàn)?new String() 不管你常量池中有沒有,我都會在堆中新建一個對象,新建出來的對象,當(dāng)然不會和其他對象相等。
那什么時候會放到字符串常量池呢,就是在使用 intern() 方法之后。 intern() 的定義:
如果當(dāng)前字符串內(nèi)容存在于字符串常量池,存在的條件是使用 equas() 方法為true,也就是內(nèi)容是一樣的,那直接返回此字符串在常量池的引用;
如果之前不在字符串常量池中,那么在常量池創(chuàng)建一個引用并且指向堆中已存在的字符串,然后返回常量池中的地址。
第一種情況,準(zhǔn)備池化的字符串與字符串常量池中的字符串有相同(equas()判斷)
String s1 = "古時的風(fēng)箏"; String a = "古時的"; String s2 = new String(a + "風(fēng)箏"); s2 = s2.intern();
這個字符串常量已經(jīng)在常量池存在了,這時,再 new 了一個新的對象 s2,并在堆中創(chuàng)建了一個相同字符串內(nèi)容的對象。
這時,s1 == s2 會返回 fasle。然后我們調(diào)用 s2 = s2.intern(),將池化操作返回的結(jié)果賦值給 s2,就會發(fā)生如下的變化。
此時,再次判斷 s1 == s2 ,就會返回 true,因?yàn)樗鼈兌贾赶蛄俗址A砍氐耐粋€字符串。
第二種情況,字符串常量池中不存在相同內(nèi)容的字符串
使用 new String() 在堆中創(chuàng)建了一個字符串對象
使用了intern()之后發(fā)生了什么呢,在常量池新增了一個對象,但是并沒有將字符串復(fù)制一份到常量池,而是直接指向了之前已經(jīng)存在于堆中的字符串對象。
因?yàn)樵?JDK 1.7 之后,字符串常量池不一定就是存字符串對象的,還有可能存儲的是一個指向堆中地址的引用,現(xiàn)在說的就是這種情況,注意了,下圖是只調(diào)用了 s2.intern(),并沒有返回給一個變量。其中字符串常量池(0x88)指向堆中字符串對象(0x99)就是intern() 的過程。
只有當(dāng)我們把 s2.intern() 的結(jié)果返回給 s2 時,s2 才真正的指向字符串常量池。
public static void main(String[] args) { String s1 = "古時的風(fēng)箏"; String s2 = "古時的風(fēng)箏"; String a = "古時的"; String s3 = new String(a + "風(fēng)箏"); String s4 = new String(a + "風(fēng)箏"); System.out.println(s1 == s2); // 【1】 true System.out.println(s2 == s3); // 【2】 false System.out.println(s3 == s4); // 【3】 false s3.intern(); System.out.println(s2 == s3); // 【4】 false s3 = s3.intern(); System.out.println(s2 == s3); // 【5】 true s4 = s4.intern(); System.out.println(s3 == s4); // 【6】 true }
① 使用字符串直接量時會在常量池創(chuàng)建對象,當(dāng)然必須是常量折疊之后的。
② 使用new String()時,new產(chǎn)生的字符串對象是位于堆中,而不是常量池中。
③ JDK7之后intern()發(fā)生過變化,現(xiàn)在如果常量池中不存在這個對像,不會復(fù)制到常量池中,而是簡單的使用堆中已有字符串對象。
④ JDK7以前的intern()不是這樣子的,以前會在常量池中創(chuàng)建一個新的對象,你可以將你的代碼,在JDK6中測試一下,結(jié)果應(yīng)該會不同。所以,你的問題不在new String()上,而是在intern()上,前者與常量池從來就沒有關(guān)系。
到此,關(guān)于“Java字符串常量池和字面量賦值的簡單介紹”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!