今天給大家?guī)淼氖窃诿嬖囍薪洺1粏柕降囊坏李}:
十年的金牛網(wǎng)站建設經驗,針對設計、前端、開發(fā)、售后、文案、推廣等六對一服務,響應快,48小時及時工作處理。網(wǎng)絡營銷推廣的優(yōu)勢是能夠根據(jù)用戶設備顯示端的尺寸不同,自動調整金牛建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調整網(wǎng)站的寬度,無論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設計,從而大程度地提升瀏覽體驗。創(chuàng)新互聯(lián)從事“金牛網(wǎng)站設計”,“金牛網(wǎng)站推廣”以來,每個客戶項目都認真落實執(zhí)行。
無論在Java還是Android中,String是一個很常見的類,但是大家真的很了解嗎,我這里有幾個題:
1.
String str1 = "abc"; String str2 = new String("abc");
這兩種創(chuàng)建String對象的方法有什么不同?
2.
String s = "a" + "b" + "c" + "d";
這里面一共創(chuàng)建了多少對象?
這兩道題昨天給筆者搞得是一臉懵逼,后來一聽這是一道很經典的面試題,就趕快查閱各種資料,現(xiàn)在已經解決了。
在解決這兩個題之前,我們先來明確幾個知識點,相信把這幾個知識點弄完之后再回頭看這兩個題,就很簡單了:
我們首先來開第一個:
引用在棧內存中存儲,對象在堆內存中存儲。
這個是我粗略畫的一張圖,這張圖可能不是很準確,但是我只想表達一個意思,我們的棧內存,存放的是我們對象的引用和我們基本數(shù)據(jù)類型的值。而堆內存中存放的是我們的對象。就是這么簡單。
為什么說String對象不可變
這里我們用一下大佬的圖:
如果用代碼表示上面的圖,那就是:
String s = "abcd"; s = s + "el";
如果沒有上面這張圖,大家可能會覺得我們只是給s對象后面加了一個el,這不還是原來的s嗎?不,我們說的String對象不可變就是這個意思,當我們給s再加上el時候,新的字符串abcdel被存放到了一個新的內存中。已經不是原來的內存了。所以說String對象不可變,如果變了,那就已經變成了一個新的對象。
那說到這里大家可能會返回去看我寫的第二個問題:按你的說法這不就7個對象嗎?abcd各占一個,每次+到一起都要重新生成對象,一共生成了7個對象。話是沒錯,但是我想說這道題說多解,等我們講完了下面這兩個知識點,這道題就完全解開了。
String創(chuàng)建對象的形式:
正如我們上面看到的,String有兩種創(chuàng)建對象的形式:
1.字面量形式:
String str = "asd";
2.標準的new形式:
String str = new String("asd");
字符串常量池的意義:
字符串常量池,又稱為字符串在字面量池。大家不要他想象的多么高深額,其實說白了他就是一塊內存,它里面存放的是我們的字符串的引用。
大家可能疑問這個東西有什么意義:假如我們要創(chuàng)建一個字符串,"a" + "b" + "c" + ....+ 我們+了一萬次,那么按照上面的說法,我們是不是為了創(chuàng)建一個很長的字符串,創(chuàng)建了很多的對象。這樣不但有的字符串被重復的創(chuàng)建了,而且占用了很多不必要的內存,代價有點大。
我們的JVM為了減少字符串的重復創(chuàng)建,維護了一個特殊的內存:就是這個字符串常量池。
在這個字符串常量池中,存放著我們字符串對象的引用。下面我們根據(jù)字符串創(chuàng)建對象的兩種形式來說明一下常量池是如何存放String對象引用的:
1.通過字面量創(chuàng)建String對象:
我們舉個例子,String str = "abc";我們通過字面量形式創(chuàng)建一個值為"abc"的對象,首先我們判斷常量池中是否存在一個引用,這個引用的值也是"abc",如果有這個值,那么我們就從常量池中拿到這個引用,返回給str。
如果沒有,那么我們就創(chuàng)建這個對象,然后把str這個引用值放到常量池中。
在這里我們要明確幾個點了:
2.通過new創(chuàng)建String對象。
這里其實只有一句話,無論常量池中是否存在相同值的引用,至少創(chuàng)建一個對象。因為不管別的,new這個語法就一定會創(chuàng)建一個新的對象,然后我們要看構造方法中的字符串,如果常量池中存在與該字符串相同值的引用,那么就不會創(chuàng)建新的對象,反之會創(chuàng)建對象之后,在常量池中存放這個引用。
這樣看起來我們的常量池的意義就很明確了,減少重復創(chuàng)建String對象,從而減少不必要的內存消耗。
如果要說他的弊端的話,應該就是犧牲了CPU的計算時間來換取空間吧,因為查找是否有相同內容的引用是CPU耗時計算,但是與前者占用內存相比,這個還是算不了什么的吧(筆者自己觀點,沒有官方支持哈哈)。
intern()方法使用
最后我們來看一下這個intern()方法,這個方法其實理解為查看常量池中是否存在對應內容的引用。如果有他會返回常量池中的引用,如果沒有則會將當前String的引用放入常量池。
我們舉幾個例子對比下就好:
public static void main(String[] args) { String str1 = "gfzy"; String str2 = str1.intern(); System.out.println(str1 == str2); }
他的結果是true。很簡單,我們首先通過字面量形式創(chuàng)建了一個“gfzy”對象,此時常量池中沒有相同內容引用,所以常量池存放str1的引用。
然后str2為str1的.intern()方法,現(xiàn)在常量池中存在“gfzy”這個內容的引用,所以我們返回這個引用,也就是str1,所以str1和str2指向同一個引用。
public static void main(String[] args) { String str1 = new String("gfzy"); String str2 = "gfzy"; System.out.println(str1 == str2); }
這個結果為false,首先我們看到str1是new了一個對象,所以他肯定是在堆內存中一塊新的內存,而構造方法中的“gfzy”,首先在常量池中會拿到他的引用,然后返回這個引用。
str2直接拿到了常量池中的引用,所以一個是堆內存新創(chuàng)建的,一個是原來常量池中的引用,兩者指向不是一個內存,所以是false。
而如果是這樣呢:
public static void main(String[] args) { String str1 = new String("gfzy").intern(); String str2 = "gfzy"; System.out.println(str1 == str2); }
這個結果為true。在原來的基礎上只是添加了一個inter方法,在new String("gfzy");時候,現(xiàn)在常量池中已經有了這個引用。而現(xiàn)在intern,我們會拿到常量池中的引用,所以str1為常量池中的引用,str2和上面一樣,所以返回true。
現(xiàn)在這幾個問題都解決完了,我們再返回頭看之前的兩個問題,第一個問題就很簡單了,而我們剛才也說過了。
第二個是看常量池中是否已經存在了對應的引用,如果沒有,那么是"a","b","ab","c","abc","d","abcd"七個對象,如果常量池中存在引用,那么只創(chuàng)建了一個“abcd”(這個是編譯器優(yōu)化后的結果,其實筆者這里也是有點懵逼,個人感覺應該是3個額,希望這個問題有讀者多多留言,幫我解答一下這個問題)。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對創(chuàng)新互聯(lián)的支持。如果你想了解更多相關內容請查看下面相關鏈接