說到內存管理,筆者這里想先比較一下java與c、c++之間的區別:
在c、c++中,內存管理是由程序員負責的,也就是說程序員既要完成繁重的代碼編寫工作又要時??紤]到系統內存的維護
在java中,程序員無需考慮內存的控制和維護,而是交由jvm自動管理,這樣就不容易出現內存泄漏和溢出的問題。然而,一旦出現內存泄漏和溢出方面的問題,如果不了解jvm的內存管理機制就很難找到錯誤所在。
1.jvm運行時數據區
jvm在運行java程序的時候會將它所管理的內存劃分為若干個不同的區域,這些區域不僅有自己的用途,還有創建和銷毀的時間。一般來說包含以下幾個運行時數據區:
其中的橙色區域是各個線程私有的,即每個線程都會有自己的一份,而綠色區域是各個線程共享的。
2.java對象的創建
類加載檢查
當jvm掃描到new關鍵字時,首先會去檢查這個指令的參數是否能夠在常量池中定位到一個類的符號引用,并且檢查這個類的符號引用代表的類是否已被加載、解析和初始化過。如果沒有,就必須先執行相應的類加載過程。
內存分配
當類加載檢查通過后,jvm需要為新生對象分配內存,即是把一塊確定大小的內存從java堆中劃分出來。常用的劃分方法有兩種:指針碰撞(要求堆內存絕對規整)、空閑列表(堆內存并不規整)。
內存初始化
jvm需要將分配到的內存空間都初始化為零值(不包括對象頭),這就保證了對象的實例字段在java代碼中可以不賦初始值就直接使用,也就是說程序能夠訪問到這些字段的數據類型所對應的零值。
對象初始化
執行<init>方法,將對象按照程序員的意愿進行初始化。
3.對象的訪問定位
對象創建好了,我們還希望能夠快速的訪問到這些對象,這就需要jvm棧上的reference(引用)數據來找到堆中的具體對象,而目前使用最多的訪問方式有“句柄方式”和“直接指針”兩種。
使用句柄方式訪問的話,就需要在堆中劃分一部分內存來作為句柄池,reference變量中存儲的就是對象的句柄地址,而句柄中包含了對象實例數據和類型數據各自的具體地址信息。
使用直接指針訪問的話,reference變量中存儲的直接就是對象地址,但是需要考慮如何放置類型數據的相關信息。
jvm運行時數據區除了pc寄存器之外,其他的內存區域都有可能發生內存溢出的異常情況。pc寄存器是唯一一個在jvm規范中沒有規定任何outofmemoryerror(oom)情況的區域。
堆溢出
java中的堆用于存儲對象實例,如果不斷地創建對象,并且保證gc roots到對象之間有可達路徑以避免gc的回收處理,那么在對象的數量達到最大堆的容量限制后就會發生堆溢出的異常情況。
棧溢出(包括jvm棧和本地方法棧)
1.如果線程請求的棧深度大于jvm所允許的最大深度,將拋出stackoverflowerror異常;
2.如果jvm在擴展棧時無法申請到足夠的內存空間,將拋出outofmemoryerror異常。
此外,還有方法區溢出、常量池溢出、本機內存溢出等等。
以上這篇淺談java內存管理與內存溢出異常就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持服務器之家。
原文鏈接:http://www.cnblogs.com/Wilange/archive/2017/10/17/7654964.html