一、前言
若想自己編寫的Java程序高效運行,以及進行正確、高效的異常診斷,JVM是不得不談的一個話題。本”JVM進階“專欄大部分內容均來源于經典書籍《深入理解Java虛擬機》。
二、棧存儲
言歸正傳,本文重點從虛擬機內存模型(運行時數據區域)入手。先看下圖:
這是一張比較官方的虛擬機模型圖,今天講的就是虛線框中棧的部分。
棧是我們最常用的內存區域。它主要用來存放基本類型變量,局部變量以及對象的引用。例如:User user = new User();這里的user就是對象的引用也可以理解為地址,指引著虛擬機要去哪里找user這個對象。 他們的基本關系如圖:
由上圖可知,當我們將一個對象作為方法的參數時,我們在方法中改變對象的值,也會影響到原來對象的值,因為我們只是改變了圖中內存區域的值,他的指引(地址)還是一樣的。同時也可以看出,棧的內存區域是連續的,有大小限制的,如果超過了就會拋出棧溢出的異常StackOverflowError。
在每個方法執行的時候,都會創建一個個的棧幀,用于保存局部變量表,操作數棧,動態鏈接等信息(以后都會詳細講解)。每次方法的調用都會對應著一個棧幀,因此可以解釋當我們在寫遞歸程序的時候會不小心報棧溢出的異常,因為棧是有限的,方法調用太多次導致棧幀堆滿了棧,所以溢出。看下面代碼:
public class Test {
private static int stackLength = 0;
private static void main(String[] args) {
try {
Test test = new Test();
test.stackOverFlow();
} catch (Throwable e) {
System.out.println("stackLength:" + stackLength);
throw e;
}
}
public void stackOverFlow() {
// 瘋狂遞歸調用
stackLength++;
stackOverFlow();
}
}
在參數-Xss128k的情況下的報錯。(eclipse中設置參數:右鍵代碼選擇Run As-->Run Configurations,在Arguments欄下的VM arguments中填入參數,再Apply,再run)
每次在方法執行完畢的時候,虛擬機會自動釋放掉為該棧所分配的空間,在棧中,對應著一個棧幀的出棧。虛擬機會自動分配與回收內存,因此效率比較高。
三、總結
最后做一下棧的總結:
存放基本類型變量,局部變量,對象的引用;
系統自動分配與回收內存,效率較高,快速,存取速度比堆要快;
是一塊連續的內存的區域,有大小限制,如果超過了就會棧溢出,并拋出棧溢出的異常StackOverflowError;
Java會自動釋放掉為該變量所分配的內存空間;
棧又分為java棧和本地方法棧。顧名思義,本地方法棧自然就是為本地方法提供服務的,java棧是為java服務的。
注意??:JVM棧是每個線程私有的!
四、拓展閱讀
《JVM虛擬機專欄》
審核編輯:符乾江
-
JAVA
+關注
關注
19文章
2957瀏覽量
104544 -
虛擬機
+關注
關注
1文章
908瀏覽量
28086
發布評論請先 登錄
相關推薦
評論