0.寫在前面

這兩天做了一道常見(jiàn)的Java面試題,毫無(wú)懸念的做錯(cuò)了,在運(yùn)行出正確答案之后,發(fā)現(xiàn)以自己的知識(shí)儲(chǔ)備竟然無(wú)法完整的解釋為什么,十分慚愧,于是有了這篇文章,對(duì)其進(jìn)行總結(jié)反思。

先看下題目:

運(yùn)行結(jié)果為:
hello hello changed
后面兩個(gè)還能理解,形參、實(shí)參、值傳遞、引用傳遞啥的一混合,還能說(shuō)得過(guò)去,可是第一個(gè)為什么是hello呢,str不是被重新賦值了嗎,怎么打印的還是原來(lái)的值。
在經(jīng)歷了上面的疑惑之后,一頓百度,額不對(duì),谷歌之后,發(fā)現(xiàn)對(duì)下面這些概念了解的還不是很透徹:
什么是棧內(nèi)存、堆內(nèi)存,它們有什么區(qū)別?
初始化一個(gè)基本類型數(shù)據(jù)或者一個(gè)對(duì)象在內(nèi)存中是如何進(jìn)行的?
什么是值傳遞、引用傳遞,它們有什么區(qū)別?
String類型的數(shù)據(jù)存放在內(nèi)存的什么區(qū)域?
String str = “a”; 和 String str = new String("a"); 在內(nèi)存分配上有什么區(qū)別?
帶著這些疑問(wèn),一起往下看。
棧內(nèi)存(stack)
在函數(shù)中定義的一些基本類型的變量(byte、short、int、long、float、double、boolean、char)和對(duì)象的引用變量(Object obj = new Object(); obj為引用變量)都在函數(shù)的棧內(nèi)存中分配。
當(dāng)在一段代碼塊中定義一個(gè)變量時(shí),Java就在棧中為這個(gè)變量分配內(nèi)存空間,當(dāng)超過(guò)變量的作用域后,Java會(huì)自動(dòng)釋放掉為該變量所分配的內(nèi)存空間,該內(nèi)存空間可以立即被另作他用。
棧內(nèi)存的優(yōu)勢(shì)是,存取速度比堆要快,僅次于寄存器,棧內(nèi)存數(shù)據(jù)可以共享。但缺點(diǎn)是,存在棧中的數(shù)據(jù)大小與生存期必須是確定的,缺乏靈活性。
堆內(nèi)存(heap)
由new創(chuàng)建的對(duì)象和數(shù)組(數(shù)組new不new都可以)存放在堆內(nèi)存中,堆中分配的內(nèi)存由JVM垃圾回收機(jī)制進(jìn)行管理。
在堆內(nèi)存中存儲(chǔ)的對(duì)象或數(shù)組,可以在棧內(nèi)存中對(duì)應(yīng)一個(gè)引用變量,引用變量的取值為對(duì)象或數(shù)組在堆內(nèi)存中的首地址,程序可以通過(guò)棧內(nèi)存的引用變量來(lái)對(duì)數(shù)組或?qū)ο筮M(jìn)行操作。
Object obj = new Object(); obj為引用變量,可以通過(guò)obj變量操作Object。
基本類型數(shù)據(jù)
int?a?=?1; int?b?=?1; int?c?=?2;

變量
步驟分析:
1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為a的引用,然后查找棧內(nèi)存中是否存在1這個(gè)值,未找到,將1存入棧內(nèi)存并將變量a指向1。
2.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為b的引用,然后查找棧內(nèi)存中是否存在1這個(gè)值,找到了,將變量b指向1。
3.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為c的引用,然后查找棧內(nèi)存中是否存在2這個(gè)值,未找到,將2存入棧內(nèi)存并將變量c指向2。
在上述步驟可以看到,棧內(nèi)存中的數(shù)據(jù)是可以共享的,雖然數(shù)據(jù)是共享的,但是變量b的修改,并不會(huì)影響到變量a。
對(duì)象
Object?obj?=?new?Object();

對(duì)象
步驟分析:
1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為obj的引用。
2.在堆內(nèi)存中創(chuàng)建一個(gè)Object對(duì)象,堆內(nèi)存會(huì)自動(dòng)計(jì)算Object對(duì)象的首地址值,假設(shè)為0x0001。
3.棧內(nèi)存中的變量obj指向堆內(nèi)存中Object對(duì)象的首地址0x0001。
先來(lái)說(shuō)說(shuō)形參和實(shí)參,看下面一段代碼:
public?class?Test?{
?public?static?void?main(String[]?args)?{
?int?a?=?0;
?fun(a);
?}
?public?static?void?fun(int?a)?{
?a?=?1;
?}
}其中,fun(int a)中的a,它只有在fun方法被調(diào)用期間a才有意義,也就是會(huì)被分配內(nèi)存空間,在fun方法執(zhí)行完成后,a就會(huì)被銷毀釋放空間,這個(gè)a為?形參?。
fun(a)中的a,它是方法被調(diào)用前就已經(jīng)被初始化了的,并且在方法被調(diào)用時(shí)傳入,這個(gè)a為?實(shí)參?。
了解完了形參和實(shí)參,我們?cè)賮?lái)說(shuō)值傳遞和引用傳遞:
值傳遞:
在方法的執(zhí)行過(guò)程中,實(shí)參把它的實(shí)際值傳遞給形參,此傳遞過(guò)程就是將實(shí)參的值復(fù)制一份傳遞到函數(shù)中,這樣如果在函數(shù)中對(duì)該值(形參的值)進(jìn)行了操作,將不會(huì)影響實(shí)參的值。
java的八種基本數(shù)據(jù)類型或者String、Integer、Double作為參數(shù)時(shí),可以理解為值傳遞,修改形參不會(huì)影響實(shí)參。
引用傳遞:
在方法的執(zhí)行過(guò)程中,形參和實(shí)參的內(nèi)容相同,指向堆內(nèi)存中的同一塊內(nèi)存地址,也就是說(shuō)操作的其實(shí)都是源數(shù)據(jù),修改形參會(huì)影響實(shí)參。
java的類類型、接口類型和數(shù)組作為參數(shù)時(shí),可以理解為引用傳遞,修改形參會(huì)影響實(shí)參。
5.String類型
String類型十分特殊,它不屬于基本數(shù)據(jù)類型,但又可以像基本數(shù)據(jù)類型一樣用?=?賦值,還可以通過(guò)?new?進(jìn)行創(chuàng)建,一起來(lái)看看兩種創(chuàng)建方式在內(nèi)存中有什么區(qū)別。
String str = “a”;

String str = “a”;
步驟分析:
1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為str的引用。
2.在常量池中查找是否有字符串a(chǎn),沒(méi)有找到,創(chuàng)建一個(gè)字符串a(chǎn)。
3.棧內(nèi)存中的變量str指向常量池中的字符串a(chǎn)。
String str = new String("a");

String str = new String("a");
步驟分析:
1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為str的引用。
2.在堆內(nèi)存中創(chuàng)建一個(gè)String對(duì)象,堆內(nèi)存會(huì)自動(dòng)計(jì)算String對(duì)象的首地址值,假設(shè)為0x0001。
3.棧內(nèi)存中變量str指向堆內(nèi)存中String對(duì)象的首地址0x0001。
4.String對(duì)象首先到常量池中查找有沒(méi)有字符串a(chǎn),如果有則指向字符串a(chǎn),如果沒(méi)有則創(chuàng)建。
在學(xué)習(xí)了上面的知識(shí)之后,我們?cè)倩剡^(guò)頭來(lái)分析一下這道面試題:

以A、B、C標(biāo)識(shí)三段邏輯,分別來(lái)看下:
A:

步驟分析:
1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為str(實(shí)參)的引用。
2.在常量池中查找字符串hello,沒(méi)有找到,創(chuàng)建一個(gè)字符串hello。
3.棧內(nèi)存中的變量str(實(shí)參)指向常量池中的字符串hello。
4.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為str(形參)的引用。
5.在常量池中查找字符串changed,沒(méi)有找到,創(chuàng)建一個(gè)字符串changed。
6.棧內(nèi)存中的變量str(形參)指向常量池中的字符串changed。
此時(shí)打印實(shí)參str的值,輸出hello
B:

1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為a(實(shí)參)的引用。
2.在堆內(nèi)存中創(chuàng)建一個(gè)String對(duì)象,地址為0x0001,引用變量a(實(shí)參)指向此地址。
3.String對(duì)象首先到常量池中查找有沒(méi)有字符串hello,沒(méi)有找到,在常量池中創(chuàng)建字符串hello并指向它。
4.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為a(形參)的引用。
5.在堆內(nèi)存中創(chuàng)建一個(gè)String對(duì)象,地址為0x0011,引用變量a(形參)指向此地址。
6.String對(duì)象首先到常量池中查找有沒(méi)有字符串changed,沒(méi)有找到,在常量池中創(chuàng)建字符串changed并指向它。
此時(shí)打印實(shí)參a中的值,輸出hello
C:

1.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為a1(實(shí)參)的引用。
2.在堆內(nèi)存中創(chuàng)建一個(gè)String對(duì)象,地址為0x0001,引用變量a1(實(shí)參)指向此地址。
3.String對(duì)象首先到常量池中查找有沒(méi)有字符串hello,沒(méi)有找到,在常量池中創(chuàng)建字符串hello并指向它。
4.在棧內(nèi)存中創(chuàng)建一個(gè)變量名為a1(形參)的引用,指向0x0001地址。
5.String對(duì)象首先到常量池中查找有沒(méi)有字符串changed,沒(méi)有找到,在常量池中創(chuàng)建字符串changed并指向它,不再指向字符串hello。
此時(shí)打印實(shí)參a中的值,輸出changed
7.寫在最后
到這里,關(guān)于這道Java面試題的總結(jié)就完成了,關(guān)聯(lián)的東西還不少,如果遇到問(wèn)題或者有錯(cuò)誤的地方,可以給我留言,謝謝!
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。
名稱欄目:面試常問(wèn)的一道Java線程問(wèn)題、從而引發(fā)的連環(huán)慘案-創(chuàng)新互聯(lián)
標(biāo)題URL:http://www.chinadenli.net/article22/pdcjc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站制作、響應(yīng)式網(wǎng)站、營(yíng)銷型網(wǎng)站建設(shè)、品牌網(wǎng)站設(shè)計(jì)、網(wǎng)頁(yè)設(shè)計(jì)公司、商城網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容