Java運行時數(shù)據(jù)區(qū)域

Java虛擬機(jī)在執(zhí)行Java程序過程中會把內(nèi)存區(qū)域劃分為若干個不同的數(shù)據(jù)區(qū)域,這些區(qū)域各有各自的用途、創(chuàng)建和銷毀時間。

Java虛擬機(jī)運行時數(shù)據(jù)區(qū)

程序計數(shù)器

程序計數(shù)器占用較小的內(nèi)存空間,可以看做是當(dāng)前線程所執(zhí)行的字節(jié)碼的行號指示器,由于Java虛擬機(jī)的多線程是通過線程輪流切換并分配處理器執(zhí)行時間的方式來實現(xiàn)的,在任何一個確定的時刻,一個處理器(對于多核處理器來說就是一個內(nèi)核)都只會執(zhí)行一條線程中的指令。因此,為了線程切換后能夠恢復(fù)到正確的執(zhí)行位置,每條線程都需要有一個獨立的程序計數(shù)器。

如果線程正在執(zhí)行Java方法,則計數(shù)器記錄的是正在執(zhí)行的虛擬機(jī)字節(jié)碼指令的地址;如果正在執(zhí)行的是Native方法,則這個計數(shù)器則為空。


Java虛擬機(jī)棧

虛擬機(jī)棧也是線程私有,而且生命周期與線程相同,每個Java方法在執(zhí)行的時候都會創(chuàng)建一個棧幀(Stack Frame)用于存儲局部變量表、操作數(shù)棧、動態(tài)鏈接、方法出口等信息。

  • 局部變量表:存放了編譯器可知的各種基本數(shù)據(jù)類型(boolean、byte等)、對象引用(reference類型,它不等同于對象本身,可能是一個指向?qū)ο笃鹗嫉刂返囊弥羔槪部赡苁侵赶蛄硪粋€代表對象的句柄或其他次對象相關(guān)的位置)和returnAddress類型(指向了一條字節(jié)碼指令的地址)
局部變量表

Java虛擬機(jī)規(guī)范中,對該區(qū)域規(guī)定了這兩種異常情況:

  1. 如果線程請求的棧深度大于虛擬機(jī)所允許的深度,講拋出StackOverflowError異常;
  2. 虛擬機(jī)??梢詣討B(tài)拓展,當(dāng)擴(kuò)展時無法申請到足夠的內(nèi)存,就會拋出OutOfMemoryError異常。

本地方法棧

本地方法棧的作用與虛擬機(jī)棧作用是非常類似的。


Java堆

對大多數(shù)應(yīng)用來說,Java堆(Heap)是Java虛擬機(jī)所管理的內(nèi)存中最大的一塊,Java堆是被所有線程共享的一塊內(nèi)存區(qū)域,在虛擬機(jī)啟動時創(chuàng)建。該內(nèi)存區(qū)域唯一的目的就是存放對象實例,Java對象實例以及數(shù)組都在堆上分配(隨著JIT編譯器發(fā)展等技術(shù)成熟,所有對象分配在堆上也漸漸不是那么“絕對”了)。

Java堆是垃圾收集器管理的主要區(qū)域,因此Java堆也常被稱為“GC堆”,由于現(xiàn)在收集器基于分代收集算法,Java堆還可以細(xì)分為:新生代和老年代。

根據(jù)Java虛擬機(jī)規(guī)范的規(guī)定,Java堆可以處于物理上不連續(xù)的內(nèi)存空間中,只要邏輯上是連續(xù)的即可,就像我們的磁盤空間一樣(或者說,像鏈表一樣雖然內(nèi)存上不一定連續(xù),但邏輯上是連續(xù))。如果在堆中沒有內(nèi)存完成實例分配,而且堆也沒辦法再擴(kuò)展時,將會拋出OutOfMemoryError異常。


方法區(qū)

方法區(qū)與Java堆一樣,是各個線程共享的內(nèi)存區(qū)域,用于存儲已被虛擬機(jī)加載的類信息、常量、靜態(tài)變量、即時編譯器編譯后的代碼等數(shù)據(jù)。Java虛擬機(jī)規(guī)范對方法區(qū)的限制非常寬松,除了和Java堆一樣不需要連續(xù)的內(nèi)存和可以選擇固定大小或者可拓展外,還可以選擇不實現(xiàn)垃圾收集。相對而言,垃圾收集行為在這個區(qū)域是比較少出現(xiàn)的,但并非數(shù)據(jù)進(jìn)入了方法區(qū)就成為了永久代。該區(qū)域的內(nèi)存回收目標(biāo)主要是針對常量池的回收和對類型的卸載。

Java虛擬機(jī)規(guī)范規(guī)定,當(dāng)方法區(qū)無法滿足內(nèi)存分配需求時,講拋出OutOfMemoryError異常。


運行時常量池

運行時常量池是方法區(qū)的一部分,Class文件中除了有關(guān)類的版本、字段、方法、接口等描述信息外,還有一項信息是常量池,用于存放編譯期生成的各種字面量和符號引用,這部分內(nèi)容將在類加載后進(jìn)入方法區(qū)的運行時常量池中存放。

運行時常量池相對于Class文件常量池的另一個重要特征是具備動態(tài)性,Java語言并非不要求常量一定只有編譯期才能產(chǎn)生,也就是并非預(yù)置入Class文件中常量池的內(nèi)容才能進(jìn)入方法區(qū)運行時常量池,運行期間也可以將新的常量池放入池中。


總結(jié)

以上的數(shù)據(jù)區(qū)域,有各自的用途,也有各自創(chuàng)建和銷毀的時間,有的區(qū)域隨著虛擬機(jī)進(jìn)程的啟動而存在,有的區(qū)域則依賴用戶線程的啟動和結(jié)束而建立和銷毀。

這幾個數(shù)據(jù)區(qū)域是接下來了解Java虛擬機(jī)的基礎(chǔ)。也許一開始會看不懂,但看到以后再回來的時候?qū)⒛軌蛄私獾健?/p>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容