java把內(nèi)存分為兩種:一種是棧內(nèi)存,另一種是堆內(nèi)存。在函數(shù)中定義的一些基本類型的變量和對象的引用變量都是在函數(shù)的棧內(nèi)存中分配,當(dāng)在一段代碼塊定義一個(gè)變量時(shí),java 就在棧中為這個(gè)變量分配內(nèi)存空間,當(dāng)超過變量的作用域后(比如,在函數(shù)a中調(diào)用函數(shù)b,在函數(shù)b中定義變量a,變量a的作用域只是函數(shù)b,在函數(shù)b運(yùn)行以后,變量a會自動被銷毀。分配給它的內(nèi)存會被回收),java會自動釋放掉為該變量分配的內(nèi)存空間,該內(nèi)存空間可以立即另做他用。
堆內(nèi)存用來存放由new創(chuàng)建的內(nèi)存數(shù)組,在堆中分配的內(nèi)存,由java虛擬機(jī)的自動垃圾回收器來管理。在堆中產(chǎn)生一個(gè)數(shù)組或?qū)ο笾螅€可以在棧中定義一個(gè)特殊的變量,讓棧中的這個(gè)變量的取值等于數(shù)組或?qū)ο笤诙褍?nèi)存中的首地址,棧中的這個(gè)變量就變成了數(shù)組或?qū)ο蟮囊米兞浚院缶涂梢栽诔绦蛑惺褂脳V械淖兞縼碓L問堆中的數(shù)組或者對象,引用變量就相當(dāng)于為數(shù)組或者對象起的一個(gè)名字。引用變量是普通的變量,定義時(shí)在棧中分配,引用變量在程序運(yùn)行到其他作用域之外后邊釋放。而數(shù)組和對象本省在堆中分配,即使程序運(yùn)行到使用new產(chǎn)生的數(shù)組或者對象的語句所在的代碼塊之外,數(shù)組和對象本省占據(jù)的內(nèi)存不會被釋放。數(shù)組和對象在沒有引用變量指向它的時(shí)候,才變?yōu)槔荒茉俦皇褂茫陔S后的一個(gè)不確定時(shí)間被垃圾回收器收走(釋放掉)。這也是java比較占內(nèi)存的原因,實(shí)際上,棧中的變量指向堆內(nèi)存中的變量,這就是java中的指針。
代碼實(shí)例demo1:單個(gè)對象創(chuàng)建
1
2
3
4
5
6
7
8
9
10
11
12
|
class person { string name ; int age ; public void tell() { system.out.println( "姓名:" +name+ ",年齡:" +age); } } public class demo1 { public static void main(string[] args) { person per = new person() ; } } |
在上述程序中實(shí)例化了一個(gè)對象per,在實(shí)例化的過程中需要再內(nèi)存中開辟空間,這其中就包括棧內(nèi)存和堆內(nèi)存,具體的內(nèi)存分配如下圖所示:
圖1-1 對象的實(shí)例化過程
我們可以從上圖中發(fā)現(xiàn),對象名稱per被保存在了棧內(nèi)存中(更加準(zhǔn)確的說法是,在棧內(nèi)存中保存的是堆內(nèi)存空間的訪問地址),而對象的具體內(nèi)容,比如屬性name和age,被保存在堆內(nèi)存中。因?yàn)閜er對象只是被實(shí)例化,還沒有被具體賦值,所以都是默認(rèn)值。字符串的默認(rèn)值為null,int的類型的默認(rèn)值為0。前面已經(jīng)提到,堆內(nèi)存空間必須使用new關(guān)鍵字才能開辟。
代碼實(shí)例demo2:多個(gè)對象創(chuàng)建
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
class person { string name ; int age ; public void tell() { system.out.println( "姓名:" +name+ ",年齡:" +age); } } public class demo2 { public static void main(string[] args) { person per1 = new person() ; person per2 = new person() ; per1.name= "張三" ; per1.age= 30 ; per2.age= 33 ; per1.tell(); per2.tell(); } } |
圖1-2 實(shí)例化兩個(gè)對象
關(guān)鍵概念:類跟數(shù)組一樣,都是屬于引用類型,引用類型就是指同一個(gè)堆內(nèi)存可以被多個(gè)棧內(nèi)存指向,下面來看一下引用傳遞的簡單實(shí)例。
代碼實(shí)例demo3:對象引用傳遞1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
class person { string name ; int age ; public void tell() { system.out.println( "姓名:" +name+ ",年齡:" +age); } } public class demo3 { public static void main(string[] args) { person per1 = new person() ; person per2 = per1 ; //-------注意-------- per1.name= "張三" ; per1.age= 30 ; per2.age= 33 ; per1.tell(); per2.tell(); } } |
程序運(yùn)行結(jié)果為:
從程序的運(yùn)行結(jié)果可以發(fā)現(xiàn),兩個(gè)對象輸出的內(nèi)容一樣,實(shí)際上所謂的引用傳遞,就是將一個(gè)堆內(nèi)存空間的使用權(quán)交給多個(gè)棧內(nèi)存空間,每個(gè)棧內(nèi)存空間都可以修改堆內(nèi)存空間的內(nèi)容,此程序的內(nèi)存分配圖如下所示:
圖1-3 對象引用的傳遞內(nèi)存分配
圖1-3 對象引用的傳遞內(nèi)存分配(續(xù))
注意:上述實(shí)例中對象per2沒有堆內(nèi)存空間,這是因?yàn)閷ο髉er2只進(jìn)行聲明操作,也沒有進(jìn)行實(shí)例化操作。只是使用new關(guān)鍵字,實(shí)例化以后才會有堆內(nèi)存空間
代碼實(shí)例demo4:對象引用傳遞2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
class person { string name ; int age ; public void tell() { system.out.println( "姓名:" +name+ ",年齡:" +age); } } public class demo4 { public static void main(string[] args) { person per1 = new person() ; person per2 = new person() ; per1.name= "張三" ; per1.age= 30 ; per2.name= "李四" ; per2.age= 33 ; per2=per1 ; //-----注意---- per1.tell(); per2.tell(); } } |
上述運(yùn)行程序結(jié)果為:
從程序的輸出結(jié)果可以發(fā)現(xiàn)跟demo3差不多。不過內(nèi)存分配發(fā)生了一些變化,具體如下所示:
圖1-4 (垃圾對象)的產(chǎn)生
注意點(diǎn):
1.java本身提供垃圾收集機(jī)制(garbage collection,gc),會不定期釋放不用的內(nèi)存空間,只要對象不用了,就會等待gc釋放空間,如上面堆內(nèi)存中的name="李四";age=33。
2.一個(gè)棧內(nèi)存只能指向一個(gè)堆內(nèi)存空間,如果要想指向其他堆內(nèi)存空間,則必須先斷開已有的指向,才能分配新的指向。
java中常見的內(nèi)存區(qū)域
在java中主要存在4塊內(nèi)存空間,這些內(nèi)存的名稱及作用如下:
1.棧內(nèi)存空間:保存所有對象的名稱。
2.堆內(nèi)存空間:保存每個(gè)對象的具體屬性內(nèi)容。
3.全局?jǐn)?shù)據(jù)區(qū):保存static類型的屬性值。
4.全局代碼區(qū):保存所有的方法定義。
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時(shí)也希望多多支持服務(wù)器之家!
原文鏈接:http://www.cnblogs.com/js2ja/p/6564790.html