JVM內(nèi)存模型
之前看過(guò)很多關(guān)于jvm內(nèi)存模型的文章,什么有堆棧辛润,方法區(qū),程序計(jì)數(shù)器等劃分见秤,也有文章說(shuō)主存和工作內(nèi)存的模型砂竖。
這里我覺(jué)得無(wú)論是jvm內(nèi)存區(qū)域的劃分,還是主存工作內(nèi)存的使用鹃答,都是屬于jvm內(nèi)存模型內(nèi)的知識(shí)乎澄。基于自己的理解测摔,簡(jiǎn)單總結(jié)一下置济,單純只是個(gè)人總結(jié),不一定是正確的锋八,假如看到有錯(cuò)誤的地方可以幫忙指出浙于。
jvm運(yùn)行期的內(nèi)存區(qū)域劃分:
上述圖可以比較直觀的看出jvm運(yùn)行期,哪些內(nèi)存區(qū)域?qū)儆诠蚕淼男矗男儆诰€程私有的羞酗。下面簡(jiǎn)單的介紹每個(gè)內(nèi)存區(qū)域:
1.線程私有的內(nèi)存區(qū)域:
(1)java 棧(jvm stack)
主要存放一個(gè)個(gè)棧幀,每個(gè)棧幀對(duì)應(yīng)著線程執(zhí)行每個(gè)方法相關(guān)的數(shù)據(jù)(局部變量表紊服,操作棧檀轨,動(dòng)態(tài)鏈接胸竞,方法出口等信息),每一個(gè)方法執(zhí)行完了裤园,對(duì)應(yīng)的棧幀就會(huì)從java棧出棧撤师。當(dāng)棧的深度達(dá)到j(luò)vm所允許的最大深度,就會(huì)拋出 stackoverflowerror 的錯(cuò)誤拧揽,當(dāng)擴(kuò)展無(wú)法申請(qǐng)到足夠的內(nèi)存則會(huì)拋出 outofmemoryerror 的錯(cuò)誤剃盾。
(2)程序計(jì)數(shù)器(PC register)
很小的內(nèi)存空間,因?yàn)閖vm底層執(zhí)行程序是執(zhí)行字節(jié)碼指令淤袜,而程序計(jì)數(shù)器就記錄著當(dāng)前指令的執(zhí)行位置痒谴。隨著指令執(zhí)行而變化,從而獲取下一個(gè)需要執(zhí)行的指令(可實(shí)現(xiàn)分支铡羡,循環(huán)积蔚,跳轉(zhuǎn),異常)烦周。
(3)本地方法棧(native method stack)
與java棧存儲(chǔ)內(nèi)容很相似尽爆,只是java棧是服務(wù)于java方法(字節(jié)碼),而本地方法棧是服務(wù)于native方法读慎。同樣也會(huì)拋出上面提到的異常錯(cuò)誤漱贱。
2.線程共享的內(nèi)存區(qū)域:
(1)堆(heap)
這一塊主要是存放對(duì)象數(shù)據(jù),是jvm內(nèi)存中最大的一塊夭委。包含了新生代和年老代幅狮,也是GC(垃圾回收)主要收集的區(qū)域。(GC算法會(huì)在后面的文章里提到)株灸,當(dāng)內(nèi)存申請(qǐng)超過(guò)了jvm定義的內(nèi)存大小崇摄,則會(huì)報(bào)outofmemory的錯(cuò)誤。
(2.1)本地方法區(qū)(Method Area)
主要存放的是類信息慌烧,常量逐抑,靜態(tài)變量等數(shù)據(jù),在jdk1.8之前屹蚊,hotspot里是將方法區(qū)當(dāng)做永久代來(lái)使用泵肄,之后廢除了永久代,這里會(huì)在另外一篇文章里解釋淑翼。
(2.2)運(yùn)行時(shí)常量池(Runtime Constant Pool)
屬于方法區(qū)的一部分,用于存放編譯期生成的各種字面量和符號(hào)引用品追,如字符串玄括,final變量,類名和方法名常量等(這里需要注意常量池的常量的存儲(chǔ)大腥馔摺)遭京。