Java虛擬機(jī)內(nèi)存結(jié)構(gòu)
時(shí)間:20170227
Java與C++之間有一堵有內(nèi)存動(dòng)態(tài)分配和垃圾收集技術(shù)所圍成的“高墻”,墻外面的人想進(jìn)來(lái),墻里面的人卻想出去。
線程:順序執(zhí)行單元。
一侧馅、運(yùn)行時(shí)數(shù)據(jù)區(qū)
java運(yùn)行時(shí)數(shù)據(jù)區(qū)圖如下:
1線程獨(dú)占區(qū)
1.1.程序計(jì)數(shù)器
是一塊較小的內(nèi)存空間,記錄當(dāng)前線程所執(zhí)行的字節(jié)碼行號(hào)呐萌。
1.2.java虛擬機(jī)棧
存放方法運(yùn)行時(shí)所需的數(shù)據(jù)馁痴。
虛擬機(jī)棧:為虛擬機(jī)執(zhí)行Java方法服務(wù)。
虛擬機(jī)棧描述的是Java方法執(zhí)行的內(nèi)存模型:每個(gè)方法在執(zhí)行的同時(shí)都會(huì)創(chuàng)建一個(gè)棧幀用于存儲(chǔ)局部變量表肺孤、操作數(shù)棧罗晕、動(dòng)態(tài)鏈接、方法出口赠堵。每一個(gè)方法從調(diào)用直至執(zhí)行完成的過(guò)程小渊,就對(duì)應(yīng)著一個(gè)棧幀在虛擬機(jī)中入棧到出棧的過(guò)程。
同程序計(jì)數(shù)器一樣茫叭,java虛擬機(jī)棧也是線程獨(dú)占的酬屉。他的生命周期與線程相同。
-
棧幀
- 每個(gè)方法執(zhí)行杂靶,都會(huì)創(chuàng)建一個(gè)棧幀梆惯,伴隨著方法從創(chuàng)建到執(zhí)行完成。用于存儲(chǔ)局部變量表吗垮,操作數(shù)棧垛吗、動(dòng)態(tài)鏈接、方法出口等烁登。
-
局部變量表
- 用于存放編譯器可知的各種基本數(shù)據(jù)類型怯屉,引用類型、returnAddress類型饵沧。
- 局部變量表的內(nèi)存空間在編譯期完成分配锨络,當(dāng)進(jìn)入一個(gè)方法時(shí),這個(gè)方法需要在棧幀中分配多少內(nèi)存時(shí)固定的狼牺,在方法運(yùn)行期間時(shí)不會(huì)改變局部變量表的大小的羡儿。
-
虛擬機(jī)棧大小及棧幀大小
- StackOverflowError產(chǎn)生的原因分析:由于每個(gè)方法在執(zhí)行的過(guò)程中都會(huì)在虛擬機(jī)棧中創(chuàng)建一個(gè)棧幀,當(dāng)出現(xiàn)循環(huán)調(diào)用時(shí)(例如A方法不停的調(diào)用自身)是钥,就會(huì)不停的生成棧幀掠归,并壓入虛擬機(jī)棧中,直到超過(guò)棧的大小悄泥,拋出錯(cuò)誤虏冻。StackOverflowError。
- OutOfMemory:為上述情況極端情形弹囚。
1.3.本地方法棧
為JVM所調(diào)用的Native方法服務(wù)厨相。
本地方法棧與虛擬機(jī)棧的作用相似。
虛擬機(jī)棧:為虛擬機(jī)執(zhí)行Java方法服務(wù)。
本地方法棧:為JVM所調(diào)用的Native方法服務(wù)蛮穿。
2線程獨(dú)占區(qū)
2.1 方法區(qū)
存儲(chǔ)運(yùn)行時(shí)常量池庶骄,已被虛擬機(jī)加載的類信息、常量践磅、靜態(tài)變量瓢姻、即時(shí)編譯器編譯后的代碼等數(shù)據(jù)。
方法區(qū)與java堆一樣是各個(gè)線程共享的內(nèi)存區(qū)域音诈。
- 類信息:包括類的版本、字段绎狭、方法细溅、接口
2.2 java堆
存儲(chǔ)對(duì)象實(shí)例
java堆是被所有線程所共享的一塊內(nèi)存區(qū)域,在虛擬機(jī)啟動(dòng)時(shí)創(chuàng)建儡嘶。在內(nèi)存區(qū)域的唯一目的就是存放對(duì)象的實(shí)例喇聊,幾乎所有的對(duì)象實(shí)例都在這里分配內(nèi)存。
對(duì)于大多數(shù)應(yīng)用來(lái)說(shuō)蹦狂,java堆是java虛擬機(jī)管理的內(nèi)存中最大的一塊誓篱。因?yàn)樵趈ava應(yīng)用中會(huì)不停的創(chuàng)建java對(duì)象。只要時(shí)對(duì)象的存儲(chǔ)都會(huì)放在堆中凯楔。
java堆也是垃圾收集器管理的主要區(qū)域窜骄。很多時(shí)候被成為GC堆(Garbage Collected Heap),也稱為垃圾堆。
Java堆還可以細(xì)分為:新生代和老年代摆屯。
2.3運(yùn)行時(shí)常量池
運(yùn)行時(shí)常量池(Runtime Constant Pool)是方法區(qū)的一部分邻遏。class文件中除了有類的版本、字段虐骑、方法准验、接口以外,還有一項(xiàng)信息是常量池(Constant Pool Table)廷没,用于存放編譯期生成的各種字面量和符號(hào)引用糊饱,這部分類容將在類加載后進(jìn)入方法區(qū)的運(yùn)行時(shí)常量池中存放。
2.4直接內(nèi)存
直接內(nèi)存并不是虛擬機(jī)運(yùn)行時(shí)數(shù)據(jù)區(qū)的一部分颠黎,也不是java虛擬機(jī)規(guī)范中定義的內(nèi)存區(qū)域另锋。