一、Java內存區域
再來個總結的表,然後一一再細說:
內存區域 | 線程隔離 | 作用 | 配置參數 | 異常 | 其他說明 |
---|---|---|---|---|---|
程序計數器 | 隔離 | 字節碼行號指示器 | —— | —— | |
虛擬機棧 | 隔離 | 棧幀形式存儲局部變量表、操作棧、動態鏈接、方法出口等信息,執行java方法 | -Xss |
StackOverflowError OutOfMemoryError |
|
本地方法棧 | 隔離 | 執行本地方法 | -Xoss |
StackOverflowError OutOfMemoryError |
連續內存 |
堆 | 共享 | 保存對象實例 |
-Xms -Xmx |
OutOfMemoryError |
允許不連續 |
方法區 | 共享 | 存儲類型信息、常量、靜態變量、即時編譯器編譯後的代碼緩存等數 |
-XX:PermSize:16M -XX:MaxPermSize:64M |
OutOfMemoryError | 允許不連續 |
運行時常量池 | 共享 | 存儲各種字面量與符號引用 |
允許不連續 方法區一部分 |
1.1 程序計數器
程序計數器是一塊較小的內存,屬於線程隔離區域,是當前線程所執行的字節碼的行號指示器。
字節碼解釋器工作時就是通過改變這個計數器的值來選取下一條需要執行的字節碼指令,它是程序控制流的指示器,分支、循環、跳轉、異常處理、線程恢復等基礎功能都需要依賴這個計數器來完成。
Java虛擬機的多線程是通過線程輪流切換、分配處理器執行時間的方式來實現的,在任何一個確定的時刻,一個處理器(對於多核處理器來說是一個內核)都只會執行一條線程中的指令。因此,爲了線程切換後能恢復到正確的執行位置,每條線程都需要有一個獨立的程序計數器,各條線程之間計數器互不影響,獨立存儲,我們稱這類內存區域爲“線程私有”的內存。
1.2 虛擬機棧
這塊內存區域就是我們常說的“棧”,也是線程私有或線程隔離的。生命週期與線程相同。
虛擬機棧描述的是Java方法執行的線程內存模型:每個方法被執行的時候,Java虛擬機都會同步創建一個棧幀,用於存儲局部變量表、操作數棧、動態連接、方法出口等信息。每一個方法被調用直至執行完畢的過程,就對應着一個棧幀在虛擬機棧中從入棧到出棧的過程。
經常有人把Java內存區域籠統地劃分爲堆內存(Heap)和棧內存(Stack),這種劃分方式直接繼承自傳統的C、C++程序的內存佈局結構,在Java語言裏就顯得有些粗糙了,實際的內存區域劃分要比這更復雜。不過這種劃分方式的流行也間接說明了程序員最關注的、與對象內存分配關係最密切的區域是“堆”和“棧”兩塊。其中,“堆”在稍後筆者會專門講述,而“棧”通常就是指這裏講的虛擬機棧,或者更多的情況下只是指虛擬機棧中局部變量表部分。
1.3 本地方法棧
Native Method Stacks
這塊區域和上面的虛擬機棧類似,只是虛擬機棧執行的是Java方法(也就是字節碼),而本地方法棧執行的虛擬機本地的方法。
HotSpot是將虛擬機棧和本地方法棧合併在一起。
1.4 堆
1.5 方法區
方法區和堆一樣是線程共享的。
1.6 運行時常量池
這個常量池不是一個新的區域,它只是方法區的一個部分。