本人看的深入理解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這個經典錯誤