2.2.1程序計(jì)數(shù)器
占內(nèi)存較小跺撼,可看作當(dāng)前線程執(zhí)行字節(jié)碼的行號指示器.屬于"線程私有"的內(nèi)存.當(dāng)執(zhí)行Native方法是督赤,次計(jì)數(shù)器值為空.唯一一個沒有OutOfmemoryError情況的區(qū)域.
2.2.2java虛擬機(jī)棧
線程私有,與線程生命周期相同.描述的是java方法執(zhí)行的內(nèi)存模型:每個方法執(zhí)行的同時都會創(chuàng)建一個棧幀跨扮,存儲局部變量表寒波,操作數(shù)棧吹害,動態(tài)連接均函,方法出口等信息.每個方法的調(diào)用到結(jié)束亿虽,都對應(yīng)一個棧幀入棧到出棧的過程.
局部變量表放了基本數(shù)據(jù)類型菱涤,對象引用,和returnAddress類型.其所需的內(nèi)存空間在編譯期間完成分配洛勉,運(yùn)行期間不會改變.
如果線程請求的棧深度大于虛擬機(jī)所允許的深度粘秆,拋出StackOverflowError異常.可動態(tài)擴(kuò)展,內(nèi)存溢出拋出OutOfmemoryError異常.
2.2.3本地方法棧
與虛擬機(jī)棧類似收毫,只不過是為Native方法服務(wù).HotSpot虛擬機(jī)合二為一了.
2.2.4java堆
線程共享內(nèi)存攻走,虛擬機(jī)啟動時創(chuàng)建.幾乎所有對象實(shí)例在此分配.未逃逸的對象可以分配棧里.也有更細(xì)致的分法,后面會說.無法擴(kuò)展時會拋出OutOfmemoryError異常.
2.2.5方法區(qū)
線程共享內(nèi)存區(qū)域牛哺,用于存儲已被虛擬機(jī)加載的類信息陋气,常量劳吠,靜態(tài)變量引润,即時編譯器編譯后的代碼等數(shù)據(jù).名義上與堆分開,實(shí)際上用的是堆的永久代來實(shí)現(xiàn)痒玩,方便GC管理.但是內(nèi)存更容易溢出淳附,其他虛擬機(jī)沒有.會逐漸采用Nativa Memory來實(shí)現(xiàn)方法區(qū)的規(guī)劃.jdk1.7已經(jīng)把字符串常量池移出. 方法區(qū)的回收目標(biāo)主要是針對常量池的回收和類型的卸載.
2.2.6運(yùn)行時常量池
方法區(qū)的一部分,class文件中除了有類的版本蠢古,字段奴曙,方法,接口等 還有一項(xiàng)信息是常量池草讶,用于放編譯期生成的各種字面量和符號引用洽糟,這部分內(nèi)容,將在類加載后進(jìn)入方法區(qū)的運(yùn)行時常量池.
運(yùn)行時常量池相對于class文件常量池的另外一個重要特征堕战,是具備動態(tài)特性坤溃,java并不要求常量一定只有編譯期才能產(chǎn)生,運(yùn)行期間也可能將新的常量放入池中.String的intern()方法.
2.2.7直接內(nèi)存
不是虛擬機(jī)運(yùn)行時數(shù)據(jù)區(qū)的一部分嘱丢,也不是java虛擬機(jī)規(guī)范中定義的內(nèi)存區(qū)域.nio可以使用 Native函數(shù)庫直接分配堆外內(nèi)存薪介,然后通過一個存儲在java堆的DirectByteBuffer對象作為這塊內(nèi)存的引用進(jìn)行操作.避免了在java堆和Native堆來回復(fù)制數(shù)據(jù).也可能導(dǎo)致OutOfmemoryError異常.
2.3HotSpot虛擬機(jī)對象
2.3.1對象的創(chuàng)建
虛擬機(jī)遇到new命令時,會先去檢查這個指令的參數(shù)能否在常量池中定位到一個類的符號引用越驻,并檢查這個類是否被加載汁政,解析,初始化.如果沒有就先執(zhí)行類加載.
類加載后缀旁,會為新生對象分配內(nèi)存.如果java堆的內(nèi)存是規(guī)整的记劈,使用 指針碰撞 法為對象分配內(nèi)存.如果內(nèi)存是不規(guī)整的,會有一個記錄內(nèi)存的列表并巍,然后采用 空閑列表 法為對象分配內(nèi)存. java堆內(nèi)存的是否規(guī)整取決于GC是否帶有壓縮整理功能.