java虛擬機(jī)內(nèi)存的堆區(qū)(heap),棧區(qū)(stack)和靜態(tài)區(qū)(static/method)
JAVA的JVM的內(nèi)存可分為3個(gè)區(qū):堆(heap)、棧(stack)和方法區(qū)(method)
堆區(qū):
1.存儲(chǔ)的全部是對(duì)象,每個(gè)對(duì)象都包含一個(gè)與之對(duì)應(yīng)的class的信息。(class的目的是得到操作指令)
2.jvm只有一個(gè)堆區(qū)(heap)被所有線程共享,堆中不存放基本類(lèi)型和對(duì)象引用,只存放對(duì)象本身.
3.一般由程序員分配釋放, 若程序員不釋放,程序結(jié)束時(shí)可能由OS回收 。
棧區(qū):
1.每個(gè)線程包含一個(gè)棧區(qū),棧中只保存基礎(chǔ)數(shù)據(jù)類(lèi)型的對(duì)象和自定義對(duì)象的引用(不是對(duì)象),對(duì)象都存放在堆區(qū)中
2.每個(gè)棧中的數(shù)據(jù)(原始類(lèi)型和對(duì)象引用)都是私有的,其他棧不能訪問(wèn)。
3.棧分為3個(gè)部分:基本類(lèi)型變量區(qū)、執(zhí)行環(huán)境上下文、操作指令區(qū)(存放操作指令)。
4.由編譯器自動(dòng)分配釋放 ,存放函數(shù)的參數(shù)值,局部變量的值等.
靜態(tài)區(qū)/方法區(qū):
1.方法區(qū)又叫靜態(tài)區(qū),跟堆一樣,被所有的線程共享。方法區(qū)包含所有的class和static變量。
2.方法區(qū)中包含的都是在整個(gè)程序中永遠(yuǎn)唯一的元素,如class,static變量。
3.—,全局變量和靜態(tài)變量的存儲(chǔ)是放在一塊的,初始化的全局變量和靜態(tài)變量在一塊區(qū)域, 未初始化的全局變量和未初始化的靜態(tài)變量在相鄰的另一塊區(qū)域。
代碼實(shí)例(轉(zhuǎn)自網(wǎng)絡(luò))
AppMain.java
public class AppMain //運(yùn)行時(shí), jvm 把a(bǔ)ppmain的信息都放入方法區(qū)
{
public static void main(String[] args) //main 方法本身放入方法區(qū)。
{
Sample test1 = new Sample( " 測(cè)試1 " ); //test1是引用,所以放到棧區(qū)里, Sample是自定義對(duì)象應(yīng)該放到堆里面
Sample test2 = new Sample( " 測(cè)試2 " );
test1.printName();
test2.printName();
}
}
Sample.java
public class Sample //運(yùn)行時(shí), jvm 把a(bǔ)ppmain的信息都放入方法區(qū)
{
/** 范例名稱(chēng) */
private name; //new Sample實(shí)例后, name 引用放入棧區(qū)里, name 對(duì)象放入堆里
/** 構(gòu)造方法 */
public Sample(String name)
{
this .name = name;
}
/** 輸出 */
public void printName() //print方法本身放入 方法區(qū)里。
{
System.out.println(name);
}
}
代碼的執(zhí)行過(guò)程:
堆區(qū):
1.存儲(chǔ)的全部是對(duì)象,每個(gè)對(duì)象都包含一個(gè)與之對(duì)應(yīng)的class的信息。(class的目的是得到操作指令)
2.jvm只有一個(gè)堆區(qū)(heap)被所有線程共享,堆中不存放基本類(lèi)型和對(duì)象引用,只存放對(duì)象本身.
3.一般由程序員分配釋放, 若程序員不釋放,程序結(jié)束時(shí)可能由OS回收 。
棧區(qū):
1.每個(gè)線程包含一個(gè)棧區(qū),棧中只保存基礎(chǔ)數(shù)據(jù)類(lèi)型的對(duì)象和自定義對(duì)象的引用(不是對(duì)象),對(duì)象都存放在堆區(qū)中
2.每個(gè)棧中的數(shù)據(jù)(原始類(lèi)型和對(duì)象引用)都是私有的,其他棧不能訪問(wèn)。
3.棧分為3個(gè)部分:基本類(lèi)型變量區(qū)、執(zhí)行環(huán)境上下文、操作指令區(qū)(存放操作指令)。
4.由編譯器自動(dòng)分配釋放 ,存放函數(shù)的參數(shù)值,局部變量的值等.
靜態(tài)區(qū)/方法區(qū):
1.方法區(qū)又叫靜態(tài)區(qū),跟堆一樣,被所有的線程共享。方法區(qū)包含所有的class和static變量。
2.方法區(qū)中包含的都是在整個(gè)程序中永遠(yuǎn)唯一的元素,如class,static變量。
3.—,全局變量和靜態(tài)變量的存儲(chǔ)是放在一塊的,初始化的全局變量和靜態(tài)變量在一塊區(qū)域, 未初始化的全局變量和未初始化的靜態(tài)變量在相鄰的另一塊區(qū)域。
代碼實(shí)例(轉(zhuǎn)自網(wǎng)絡(luò))
AppMain.java
public class AppMain //運(yùn)行時(shí), jvm 把a(bǔ)ppmain的信息都放入方法區(qū)
{
public static void main(String[] args) //main 方法本身放入方法區(qū)。
{
Sample test1 = new Sample( " 測(cè)試1 " ); //test1是引用,所以放到棧區(qū)里, Sample是自定義對(duì)象應(yīng)該放到堆里面
Sample test2 = new Sample( " 測(cè)試2 " );
test1.printName();
test2.printName();
}
}
Sample.java
public class Sample //運(yùn)行時(shí), jvm 把a(bǔ)ppmain的信息都放入方法區(qū)
{
/** 范例名稱(chēng) */
private name; //new Sample實(shí)例后, name 引用放入棧區(qū)里, name 對(duì)象放入堆里
/** 構(gòu)造方法 */
public Sample(String name)
{
this .name = name;
}
/** 輸出 */
public void printName() //print方法本身放入 方法區(qū)里。
{
System.out.println(name);
}
}
代碼的執(zhí)行過(guò)程:

系統(tǒng)收到了我們發(fā)出的指令,啟動(dòng)了一個(gè)Java虛擬機(jī)進(jìn)程,這個(gè)進(jìn)程首先從classpath中找到AppMain.class文件,讀取這個(gè)文件中的二進(jìn)制數(shù)據(jù),然后把Appmain類(lèi)的類(lèi)信息存放到運(yùn)行時(shí)數(shù)據(jù)區(qū)的方法區(qū)中。這一過(guò)程稱(chēng)為AppMain類(lèi)的加載過(guò)程。
接著,Java虛擬機(jī)定位到方法區(qū)中AppMain類(lèi)的Main()方法的字節(jié)碼,開(kāi)始執(zhí)行它的指令。這個(gè)main()方法的第一條語(yǔ)句就是:
Sample test1=new Sample("測(cè)試1");
語(yǔ)句很簡(jiǎn)單啦,就是讓java虛擬機(jī)創(chuàng)建一個(gè)Sample實(shí)例,并且呢,使引用變量test1引用這個(gè)實(shí)例。貌似小case一樁哦,就讓我們來(lái)跟蹤一下Java虛擬機(jī),看看它究竟是怎么來(lái)執(zhí)行這個(gè)任務(wù)的:
1、 Java虛擬機(jī)一看,不就是建立一個(gè)Sample實(shí)例嗎,簡(jiǎn)單,于是就直奔方法區(qū)而去,先找到Sample類(lèi)的類(lèi)型信息再說(shuō)。結(jié)果呢,嘿嘿,沒(méi)找到@@,這會(huì)兒的方法區(qū)里還沒(méi)有Sample類(lèi)呢。可Java虛擬機(jī)也不是一根筋的笨蛋,于是,它發(fā)揚(yáng)“自己動(dòng)手,豐衣足食”的作風(fēng),立馬加載了Sample類(lèi),把Sample類(lèi)的類(lèi)型信息存放在方法區(qū)里。
2、 好啦,資料找到了,下面就開(kāi)始干活啦。Java虛擬機(jī)做的第一件事情就是在堆區(qū)中為一個(gè)新的Sample實(shí)例分配內(nèi)存, 這個(gè)Sample實(shí)例持有著指向方法區(qū)的Sample類(lèi)的類(lèi)型信息的引用。這里所說(shuō)的引用,實(shí)際上指的是Sample類(lèi)的類(lèi)型信息在方法區(qū)中的內(nèi)存地址,其實(shí),就是有點(diǎn)類(lèi)似于C語(yǔ)言里的指針啦~~,而這個(gè)地址呢,就存放了在Sample實(shí)例的數(shù)據(jù)區(qū)里。
3、 在JAVA虛擬機(jī)進(jìn)程中,每個(gè)線程都會(huì)擁有一個(gè)方法調(diào)用棧,用來(lái)跟蹤線程運(yùn)行中一系列的方法調(diào)用過(guò)程,棧中的每一個(gè)元素就被稱(chēng)為棧幀,每當(dāng)線程調(diào)用一個(gè)方法的時(shí)候就會(huì)向方法棧壓入一個(gè)新幀。這里的幀用來(lái)存儲(chǔ)方法的參數(shù)、局部變量和運(yùn)算過(guò)程中的臨時(shí)數(shù)據(jù)。OK,原理講完了,就讓我們來(lái)繼續(xù)我們的跟蹤行動(dòng)!位于“=”前的Test1是一個(gè)在main()方法中定義的變量,可見(jiàn),它是一個(gè)局部變量,因此,它被會(huì)添加到了執(zhí)行main()方法的主線程的JAVA方法調(diào)用棧中。而“=”將把這個(gè)test1變量指向堆區(qū)中的Sample實(shí)例,也就是說(shuō),它持有指向Sample實(shí)例的引用。
OK,到這里為止呢,JAVA虛擬機(jī)就完成了這個(gè)簡(jiǎn)單語(yǔ)句的執(zhí)行任務(wù)。參考我們的行動(dòng)向?qū)D,我們終于初步摸清了JAVA虛擬機(jī)的一點(diǎn)點(diǎn)底細(xì)了,COOL!
接下來(lái),JAVA虛擬機(jī)將繼續(xù)執(zhí)行后續(xù)指令,在堆區(qū)里繼續(xù)創(chuàng)建另一個(gè)Sample實(shí)例,然后依次執(zhí)行它們的printName()方法。當(dāng)JAVA虛擬機(jī)執(zhí)行test1.printName()方法時(shí),JAVA虛擬機(jī)根據(jù)局部變量test1持有的引用,定位到堆區(qū)中的Sample實(shí)例,再根據(jù)Sample實(shí)例持有的引用,定位到方法去中Sample類(lèi)的類(lèi)型信息,從而獲得printName()方法的字節(jié)碼,接著執(zhí)行printName()方法包含的指令。
轉(zhuǎn)
http://blog.csdn.net/lyerliu/article/details/6311709接著,Java虛擬機(jī)定位到方法區(qū)中AppMain類(lèi)的Main()方法的字節(jié)碼,開(kāi)始執(zhí)行它的指令。這個(gè)main()方法的第一條語(yǔ)句就是:
Sample test1=new Sample("測(cè)試1");
語(yǔ)句很簡(jiǎn)單啦,就是讓java虛擬機(jī)創(chuàng)建一個(gè)Sample實(shí)例,并且呢,使引用變量test1引用這個(gè)實(shí)例。貌似小case一樁哦,就讓我們來(lái)跟蹤一下Java虛擬機(jī),看看它究竟是怎么來(lái)執(zhí)行這個(gè)任務(wù)的:
1、 Java虛擬機(jī)一看,不就是建立一個(gè)Sample實(shí)例嗎,簡(jiǎn)單,于是就直奔方法區(qū)而去,先找到Sample類(lèi)的類(lèi)型信息再說(shuō)。結(jié)果呢,嘿嘿,沒(méi)找到@@,這會(huì)兒的方法區(qū)里還沒(méi)有Sample類(lèi)呢。可Java虛擬機(jī)也不是一根筋的笨蛋,于是,它發(fā)揚(yáng)“自己動(dòng)手,豐衣足食”的作風(fēng),立馬加載了Sample類(lèi),把Sample類(lèi)的類(lèi)型信息存放在方法區(qū)里。
2、 好啦,資料找到了,下面就開(kāi)始干活啦。Java虛擬機(jī)做的第一件事情就是在堆區(qū)中為一個(gè)新的Sample實(shí)例分配內(nèi)存, 這個(gè)Sample實(shí)例持有著指向方法區(qū)的Sample類(lèi)的類(lèi)型信息的引用。這里所說(shuō)的引用,實(shí)際上指的是Sample類(lèi)的類(lèi)型信息在方法區(qū)中的內(nèi)存地址,其實(shí),就是有點(diǎn)類(lèi)似于C語(yǔ)言里的指針啦~~,而這個(gè)地址呢,就存放了在Sample實(shí)例的數(shù)據(jù)區(qū)里。
3、 在JAVA虛擬機(jī)進(jìn)程中,每個(gè)線程都會(huì)擁有一個(gè)方法調(diào)用棧,用來(lái)跟蹤線程運(yùn)行中一系列的方法調(diào)用過(guò)程,棧中的每一個(gè)元素就被稱(chēng)為棧幀,每當(dāng)線程調(diào)用一個(gè)方法的時(shí)候就會(huì)向方法棧壓入一個(gè)新幀。這里的幀用來(lái)存儲(chǔ)方法的參數(shù)、局部變量和運(yùn)算過(guò)程中的臨時(shí)數(shù)據(jù)。OK,原理講完了,就讓我們來(lái)繼續(xù)我們的跟蹤行動(dòng)!位于“=”前的Test1是一個(gè)在main()方法中定義的變量,可見(jiàn),它是一個(gè)局部變量,因此,它被會(huì)添加到了執(zhí)行main()方法的主線程的JAVA方法調(diào)用棧中。而“=”將把這個(gè)test1變量指向堆區(qū)中的Sample實(shí)例,也就是說(shuō),它持有指向Sample實(shí)例的引用。
OK,到這里為止呢,JAVA虛擬機(jī)就完成了這個(gè)簡(jiǎn)單語(yǔ)句的執(zhí)行任務(wù)。參考我們的行動(dòng)向?qū)D,我們終于初步摸清了JAVA虛擬機(jī)的一點(diǎn)點(diǎn)底細(xì)了,COOL!
接下來(lái),JAVA虛擬機(jī)將繼續(xù)執(zhí)行后續(xù)指令,在堆區(qū)里繼續(xù)創(chuàng)建另一個(gè)Sample實(shí)例,然后依次執(zhí)行它們的printName()方法。當(dāng)JAVA虛擬機(jī)執(zhí)行test1.printName()方法時(shí),JAVA虛擬機(jī)根據(jù)局部變量test1持有的引用,定位到堆區(qū)中的Sample實(shí)例,再根據(jù)Sample實(shí)例持有的引用,定位到方法去中Sample類(lèi)的類(lèi)型信息,從而獲得printName()方法的字節(jié)碼,接著執(zhí)行printName()方法包含的指令。
轉(zhuǎn)
posted @ 2011-07-31 14:51 水木清華77 閱讀(1485) | 評(píng)論 (0) | 編輯 收藏