這兩天看了一下深入淺出JVM這本書,推薦給高級的java程序員去看,對你了解JAVA的底層和運行機制有
比較大的幫助。
廢話不想講了.入主題:
先了解具體的概念:
JAVA的JVM的內(nèi)存可分為3個區(qū):堆(heap)、棧(stack)和方法區(qū)(method)
堆區(qū):
1.存儲的全部是對象,每個對象都包含一個與之對應的class的信息。(class的目的是得到操作指令)
2.jvm只有一個堆區(qū)(heap)被所有線程共享,堆中不存放基本類型和對象引用,只存放對象本身
棧區(qū):
1.每個線程包含一個棧區(qū),棧中只保存基礎數(shù)據(jù)類型的對象和自定義對象的引用(不是對象),對象都存放在堆區(qū)中
2.每個棧中的數(shù)據(jù)(原始類型和對象引用)都是私有的,其他棧不能訪問。
3.棧分為3個部分:基本類型變量區(qū)、執(zhí)行環(huán)境上下文、操作指令區(qū)(存放操作指令)。
方法區(qū):
1.又叫靜態(tài)區(qū),跟堆一樣,被所有的線程共享。方法區(qū)包含所有的class和static變量。
2.方法區(qū)中包含的都是在整個程序中永遠唯一的元素,如class,static變量。
為了更清楚地搞明白發(fā)生在運行時數(shù)據(jù)區(qū)里的黑幕,我們來準備2個小道具(2個非常簡單的小程序)。
AppMain.java
public class AppMain //運行時, jvm 把appmain的信息都放入方法區(qū)
{
public static void main(String[] args) //main 方法本身放入方法區(qū)。
{
Sample test1 = new Sample( " 測試1 " ); //test1是引用,所以放到棧區(qū)里, Sample是自定義對象應該放到堆里面
Sample test2 = new Sample( " 測試2 " );
test1.printName();
test2.printName();
}
}
Sample.java
public class Sample //運行時, jvm 把appmain的信息都放入方法區(qū)
{
/** 范例名稱 */
private name; //new Sample實例后, name 引用放入棧區(qū)里, name 對象放入堆里
/** 構造方法 */
public Sample(String name)
{
this .name = name;
}
/** 輸出 */
public void printName() //print方法本身放入 方法區(qū)里。
{
System.out.println(name);
}
}
OK,讓我們開始行動吧,出發(fā)指令就是:“java AppMain”,包包里帶好我們的行動向?qū)D,Let’s GO!
據(jù)區(qū).jpg)
系統(tǒng)收到了我們發(fā)出的指令,啟動了一個Java虛擬機進程,這個進程首先從classpath中找到AppMain.class文件,讀取這個文件中的二進制數(shù)據(jù),然后把Appmain類的類信息存放到運行時數(shù)據(jù)區(qū)的方法區(qū)中。這一過程稱為AppMain類的加載過程。
接著,Java虛擬機定位到方法區(qū)中AppMain類的Main()方法的字節(jié)碼,開始執(zhí)行它的指令。這個main()方法的第一條語句就是:
Sample test1=new Sample("測試1");
語句很簡單啦,就是讓java虛擬機創(chuàng)建一個Sample實例,并且呢,使引用變量test1引用這個實例。貌似小case一樁哦,就讓我們來跟蹤一下Java虛擬機,看看它究竟是怎么來執(zhí)行這個任務的:
1、 Java虛擬機一看,不就是建立一個Sample實例嗎,簡單,于是就直奔方法區(qū)而去,先找到Sample類的類型信息再說。結果呢,嘿嘿,沒找到@@,這會兒的方法區(qū)里還沒有Sample類呢。可Java虛擬機也不是一根筋的笨蛋,于是,它發(fā)揚“自己動手,豐衣足食”的作風,立馬加載了Sample類,把Sample類的類型信息存放在方法區(qū)里。
2、 好啦,資料找到了,下面就開始干活啦。Java虛擬機做的第一件事情就是在堆區(qū)中為一個新的Sample實例分配內(nèi)存, 這個Sample實例持有著指向方法區(qū)的Sample類的類型信息的引用。這里所說的引用,實際上指的是Sample類的類型信息在方法區(qū)中的內(nèi)存地址,其實,就是有點類似于C語言里的指針啦~~,而這個地址呢,就存放了在Sample實例的數(shù)據(jù)區(qū)里。
3、 在JAVA虛擬機進程中,每個線程都會擁有一個方法調(diào)用棧,用來跟蹤線程運行中一系列的方法調(diào)用過程,棧中的每一個元素就被稱為棧幀,每當線程調(diào)用一個方法的時候就會向方法棧壓入一個新幀。這里的幀用來存儲方法的參數(shù)、局部變量和運算過程中的臨時數(shù)據(jù)。OK,原理講完了,就讓我們來繼續(xù)我們的跟蹤行動!位于“=”前的Test1是一個在main()方法中定義的變量,可見,它是一個局部變量,因此,它被會添加到了執(zhí)行main()方法的主線程的JAVA方法調(diào)用棧中。而“=”將把這個test1變量指向堆區(qū)中的Sample實例,也就是說,它持有指向Sample實例的引用。
OK,到這里為止呢,JAVA虛擬機就完成了這個簡單語句的執(zhí)行任務。參考我們的行動向?qū)D,我們終于初步摸清了JAVA虛擬機的一點點底細了,COOL!
接下來,JAVA虛擬機將繼續(xù)執(zhí)行后續(xù)指令,在堆區(qū)里繼續(xù)創(chuàng)建另一個Sample實例,然后依次執(zhí)行它們的printName()方法。當JAVA虛擬機執(zhí)行test1.printName()方法時,JAVA虛擬機根據(jù)局部變量test1持有的引用,定位到堆區(qū)中的Sample實例,再根據(jù)Sample實例持有的引用,定位到方法去中Sample類的類型信息,從而獲得printName()方法的字節(jié)碼,接著執(zhí)行printName()方法包含的指令。
posted on 2008-08-21 11:00
Daniel 閱讀(5315)
評論(1) 編輯 收藏 所屬分類:
CoreJava