本人看的深入理解jvm(該版本的java se7) java運行時數據區域 Java虛擬機在執行java程式時,把記憶體劃分為幾個不同的階段,存在不同的存在時間。不同的用途 先上圖 程式計數器:是jvm中一小塊記憶體空間,可以當做當前線程的位元組碼文件的行號,位元組碼解釋器通過改變這個值來獲取下一個指令。J ...
本人看的深入理解jvm(該版本的java se7)
java運行時數據區域
Java虛擬機在執行java程式時,把記憶體劃分為幾個不同的階段,存在不同的存在時間。不同的用途
先上圖
程式計數器:是jvm中一小塊記憶體空間,可以當做當前線程的位元組碼文件的行號,位元組碼解釋器通過改變這個值來獲取下一個指令。Java多線程通過線程輪流切換處理器的執行時間來執行的,大家知道現代處理器的原理,每次一個核上只能一個線程進行運轉,由於中斷,因此每個程式計數器都是獨立的,這些程式計數器的所在的記憶體稱為線程私有的記憶體。如果線程執行的是java方法,那麼程式計數器指代的是當前位元組碼指令的地址,如果是native方法,程式計數器的值為空,這處區域是jvm沒有定義OutOfMemoryError錯誤
虛擬機棧:打個比方,一般吧java記憶體粗略分為堆棧,這個虛擬機棧中的局部變數表就是傳說中的棧,線程私有的,描述java方法執行的記憶體模型:每個方法在執行的時候都會建立一個這個;用於存儲局部變數表,操作數棧,動態鏈接,方法出口等,每一個方法從調用到執行完,就代表一個棧幀在虛擬機棧中從入棧到出棧的過程其中局部變數表存放了編譯器可知的基本變數(boolean,int,double,float,char),還有引用類型(),renturnAddress類型(指向了一個位元組碼的地址)局部變數表的大小信息在編譯期間就確定好了,運行期間不會改變其大小,該虛擬機棧回報2個錯誤,第一個就是請求的棧幀大於虛擬機棧的深度回報StackOverflowError錯誤第二個錯誤就是虛擬機棧動態擴張,擴張到無法申請到記憶體時報OutOfMemoryError錯誤
本地方法棧:和虛擬機棧差不多,不過不是java方法是native方法,java規範對native方法的語言和數據格式沒有強制要求虛擬機可以自由的去實現它,報的錯誤也是StackOverflowError錯誤和OutOfMemoryError錯誤
Java堆:第一它是所有線程共用的,虛擬機啟動創建堆,主要用於存儲對象實例和數組,Java垃圾處理器主要處理的就是java堆,因此java堆也被稱為gc堆,從記憶體回收的角度上講可以劃分為新生代和老生代,從記憶體分配角度講,可以劃分出多個線程私有的記憶體區域等,java堆有多種劃分方式,但是無論怎麼劃分都是存儲對象實例,多重劃分是為了更好的劃分記憶體和回收記憶體,java堆可以位於記憶體上不連續的空間上,只要邏輯上連續即可,java堆可以實現為固定大小的,也可以實現為可擴展大小的,要是java堆上無法為實例對象分配記憶體,那麼就會報這個OutOfMemoryError
方法區:各個線程共用的區域,主要存儲被虛擬機載入的類信息,常量,靜態變數,編譯器編譯後的代碼,和java堆一樣邏輯上記憶體,可擴展的實現大小,這個區域的垃圾回收主要針對常量池的回收和類型的卸載,其中類型的卸載要求比較高,但是必要的垃圾回收是必要的,以前低版本的就出現過嚴重的記憶體泄漏。
運行期常量池:編譯後的Class文件中的類名啊,方法名啊,常量啊被載入到虛擬機後就會放到方法去的運行常量池來保存,當然常量不止可以只是在編譯期間放到常量池中,還可以用string.intern()方法(如果常量池中存在當前字元串, 就會直接返回當前字元串. 如果常量池中沒有此字元串, 會將此字元串放入常量池中後, 再返回)來在運行期間放到常量池中,當然要是常量池中無法申請到記憶體時就會報出OutOfMemoryError這個經典錯誤