掌握了關(guān)于Java內(nèi)存的基礎(chǔ)知識(shí)之后,我們就可以對這個(gè)面試題有更深入的理解了。
1. 案例代碼
我們先編寫面試題中涉及到的代碼,如下所示:
代碼很簡單,就一行代碼!那么問題來了,這行代碼中共有幾個(gè)對象呢?1個(gè)?2個(gè)?還是3個(gè)?要想弄明白到底有幾個(gè)對象,壹哥 就給大家把這段代碼的內(nèi)存分區(qū)繪制了一下(本案例開發(fā)環(huán)境是基于JDK8)。
2. 內(nèi)存分配(重點(diǎn))
在 String s = new String("xyz"); 這行代碼中,s是String類型的變量,不是對象!‘xyz’是字符串對象,new String("xyz")也是一個(gè)對象,那么它們幾個(gè)的內(nèi)存劃分在JDK8的環(huán)境中,如下圖所示:
那么根據(jù)上圖,壹哥 就可以為各位給出本題的答案了,請仔細(xì)咯,重點(diǎn)來啦,內(nèi)存分配過程如下:
當(dāng)JVM在編譯階段加載讀取到“xyz”的時(shí)候,首先會(huì)檢查堆中的String常量池,也就是常量緩沖區(qū),檢查是否已經(jīng)有了"xyz"常量對象,如果有,則不會(huì)再次創(chuàng)建"xyz"常量對象,并直接返回該字符串的引用地址;如果沒有,則創(chuàng)建一個(gè)"xyz"常量對象,并為該對象分配一個(gè)內(nèi)存地址002返回。
當(dāng)JVM在運(yùn)行階段加載讀取到new關(guān)鍵字的時(shí)候,JVM會(huì)在堆中為其創(chuàng)建一個(gè)對象,即new String(),并為其分配內(nèi)存地址001,而堆中這個(gè)對象的內(nèi)容是上面"xyz"常量對象的引用地址002,換句話說這個(gè)堆中存的就是常量池中"xyz"的引用地址002。
最后,s 是對當(dāng)前堆中001號(hào)對象的一個(gè)地址引用,s本身不是一個(gè)對象,s只是一個(gè)String類型的變量而已!
3. 變量與對象
我們還需要弄明白一個(gè)問題,即變量與對象的區(qū)別。請問在 String s=new String("xyz");這行代碼中,s是對象嗎?
String類中有一種創(chuàng)建對象的特殊形式,就是可以使用 ""雙引號(hào) 來創(chuàng)建對象,這種創(chuàng)建方式可以在一定程度上提高程序的運(yùn)行速度,因?yàn)镴VM會(huì)自動(dòng)根據(jù)常量池中的數(shù)據(jù)存在實(shí)際情況來決定是否有必要?jiǎng)?chuàng)建新對象。在String s=new String("xyz")中,實(shí)際創(chuàng)建了2個(gè)String對象,一個(gè)是通過""雙引號(hào),在編譯期創(chuàng)建的"xyz"常量對象;另一個(gè)是通過new String(),在運(yùn)行期創(chuàng)建的。
但是 s 只是一個(gè)String類型的變量,不是對象,它只是對字符串對象的一個(gè)引用而已。所謂的對象,應(yīng)該是需要在堆中分配了內(nèi)存的類型才能叫做對象,s位于棧中,所以不是對象!
4. 問題答案(重點(diǎn))
所以String s=new String("xyz")這行代碼中究竟有幾個(gè)對象,我們的答案要分兩種情況來考慮:
●如果String常量池中已經(jīng)有了"xyz"常量對象,那么答案就是只創(chuàng)建了一個(gè)對象,即new String();
●如果String常量池中沒有創(chuàng)建"xyz"常量對象,則會(huì)創(chuàng)建兩個(gè)對象,即一個(gè)"xyz"常量對象,一個(gè)new String()對象。
更多關(guān)于“Java培訓(xùn)”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓(xùn)經(jīng)驗(yàn),課程大綱更科學(xué)更專業(yè),有針對零基礎(chǔ)的就業(yè)班,有針對想提升技術(shù)的好程序員班,高品質(zhì)課程助力你實(shí)現(xiàn)java程序員夢想。