簡單總結(jié)一下Java虛擬機運行時數(shù)據(jù)區(qū)
1. 程序計數(shù)器
是一塊較小的內(nèi)存空間,它可以看做是當前線程所執(zhí)行的字節(jié)碼的行號指示器精拟;
每條線程都需要有一個獨立的程序計數(shù)器,即“線程私有”;
2. Java虛擬機棧
Java虛擬機棧是線程私有的裕循,生命周期與線程相同;
虛擬機棧描述的是Java方法執(zhí)行的線程內(nèi)存模型:每個方法被執(zhí)行的時候涨颜,Java虛擬機都會同步創(chuàng)建一個棧幀用于存儲局部變量表费韭,操作數(shù)棧,動態(tài)連接庭瑰,方法出口等信息星持。每一個方法被調(diào)用直至執(zhí)行完畢的過程,就對應(yīng)著一個棧幀在虛擬機內(nèi)從入棧到出棧的過程弹灭。
3. 本地方法棧
4. Java堆
Java堆是虛擬機管理內(nèi)存中最大的一塊督暂,是被所有線程共享的一塊內(nèi)存區(qū)域,在虛擬機啟動時創(chuàng)建穷吮。
此內(nèi)存區(qū)域的唯一目的就是存放對象實例逻翁。
“幾乎”所有的對象實例都在堆上分配,逃逸分析 標量替換捡鱼。八回。。
Java堆是垃圾收集器管理的內(nèi)存區(qū)域驾诈;
所有線程共享的Java堆中可以劃分出多個線程私有的分配緩沖期(TLAB)缠诅,以提升對象分配的效率;
無論從什么角度乍迄,如何劃分管引,都不會改變Java堆中存儲內(nèi)容的共性,無論是哪個區(qū)域闯两,存儲的只能是對象的實例褥伴,將Java堆細分的目的只是為了更好地回收內(nèi)存,或者更快地分配內(nèi)存漾狼;
Java堆大小設(shè)置 -Xmx 和 -Xms重慢;
5. 方法區(qū)
方法區(qū)與Java堆一樣,是各個線程共享的內(nèi)存區(qū)域逊躁,它用于存儲被虛擬機加載的類型信息伤锚,常量,靜態(tài)變量,即時編譯器編譯后的代碼緩存等數(shù)據(jù)屯援;
《Java虛擬機規(guī)范》對方法區(qū)約束十分寬松猛们,除了和Java堆一樣不需要連續(xù)的內(nèi)存和可以選擇固定大小或者可擴展外,甚至還可以選擇不實現(xiàn)垃圾收集狞洋,這區(qū)域的內(nèi)存回收目標是針對常量池到回收和對類型的卸載弯淘;
6. 運行時常量池
運行時常量區(qū)是方法區(qū)的一部分。
7. 直接內(nèi)存
直接內(nèi)存并不是虛擬機運行時數(shù)據(jù)區(qū)的一部分吉懊,也不是《java虛擬機規(guī)范》中定義的內(nèi)存數(shù)據(jù)庐橙。但是這部分內(nèi)存也被頻繁地使用,而且也可能導致OOM異常出現(xiàn)借嗽。
本機直接內(nèi)存的分配不會受到Java堆大小的限制态鳖,但是,既然是內(nèi)存恶导,則肯定還是會受到本機總內(nèi)存大小以及處理器尋址空間的限制浆竭,一般服務(wù)器管理員配置虛擬機參數(shù)時,會根據(jù)實際內(nèi)存去設(shè)置-Xmx等參數(shù)信息惨寿,但經(jīng)常忽略直接內(nèi)存邦泄,使得各個內(nèi)存區(qū)域總和大于物理限制限制(包括物理的和操作系統(tǒng)級的限制),從而導致動態(tài)擴展時出現(xiàn)OOM異常裂垦。
8. HotSpot虛擬機對象探究
8.1 對象的創(chuàng)建
new 對象是如何創(chuàng)建的呢顺囊?本文討論一下普通Java對象,不包括數(shù)組和Class對象等蕉拢。
首先檢查new指令參數(shù)是否能在在常量池定位到一個類的符號引用特碳,并檢查這個符號引用代表的類是否被加載,解析和初始化過晕换。如果沒有午乓,則先執(zhí)行相應(yīng)的類加載過程。
類加載之后届巩,再分配內(nèi)存硅瞧。
“指針碰撞”:Java內(nèi)存絕對規(guī)整(所有使用過的內(nèi)存放一邊份乒,空閑的內(nèi)存放另一邊)恕汇,中間放著一個指針作為分界點的指示器,那所分配的內(nèi)存就僅僅是把那個指針向空閑空間方向移動一段與內(nèi)存大小相等的距離或辖;
“空閑列表”:內(nèi)存空間不規(guī)整瘾英,使用的內(nèi)存與空閑的內(nèi)存交錯在一起,那虛擬機會維護一個列表颂暇,記錄哪些內(nèi)存是可用的缺谴。
具體使用何種方法,有java堆是否規(guī)整決定耳鸯,而java堆是否規(guī)整則由垃圾收集器的空間壓縮能力決定湿蛔。
因此膀曾,當使用Serial,ParNew等帶壓縮整理過程的收集器時阳啥,采用指針碰撞添谊,簡單高效;
當使用CMS這種基于清除(Sweep)算法的收集時察迟,理論上就只能采用空閑列表了斩狱;
如何解決“指針碰撞”線程不安全的問題?
1. 對分配內(nèi)存的動作進行同步處理--實際上虛擬機是采用CAS配上失敗重試的方法保證更新操作的原子性扎瓶;
2.把內(nèi)存分配的動作按照線程劃分在不同的空間之中進行所踊,即每個線程在Java堆中預(yù)先分配一小塊內(nèi)存,稱為本地線程分配緩沖(TLAB)概荷,只有本地緩沖區(qū)用完了秕岛,分配新的緩沖區(qū)才需要同步鎖定,虛擬機是否使用TLAB乍赫,可以通過-XX:+/-UseTLAB參數(shù)決定瓣蛀;
內(nèi)存分配完,虛擬機將分配的內(nèi)存空間(但不包括對象頭)都初始化為零值(如果使用了TLAB雷厂,這項工作也可以提前至TLAB分配時提前執(zhí)行)惋增;
然后虛擬機對對象進行必要的設(shè)置,這些信息放在對象頭中改鲫;