Java虛擬機(jī)在執(zhí)行Java程序的過程中會(huì)將其管理的內(nèi)存劃分為若干個(gè)不同的數(shù)據(jù)區(qū)域,這些區(qū)域有各自的用途、創(chuàng)建和銷毀的時(shí)間,有些區(qū)域隨虛擬機(jī)進(jìn)程的啟動(dòng)而存在,有些區(qū)域則是依賴用戶線程的啟動(dòng)和結(jié)束來建立和銷毀。Java虛擬機(jī)所管理的內(nèi)存包括以下幾個(gè)運(yùn)行時(shí)數(shù)據(jù)區(qū)域,如圖:

- 程序計(jì)數(shù)器(線程私有):指向當(dāng)前線程正在執(zhí)行的字節(jié)碼指令
- 虛擬機(jī)棧(線程私有): Java方法執(zhí)行的內(nèi)存模型。每個(gè)方法被執(zhí)行的時(shí)候,都會(huì)創(chuàng)建一個(gè)棧幀,把棧幀壓入棧,當(dāng)方法正常返回或者拋出未捕獲的異常時(shí),棧幀就會(huì)出棧。
棧幀:棧幀存儲(chǔ)方法的相關(guān)信息,包含局部變量數(shù)表、返回值、操作數(shù)棧、動(dòng)態(tài)鏈接
a. 局部變量表:包含了方法執(zhí)行過程中的所有變量。局部變量數(shù)組所需要的空間在編譯期間完成分配,在
方法運(yùn)行期間不會(huì)改變局部變量數(shù)組的大小。
b. 返回值:如果有返回值的話,壓入調(diào)用者棧幀中的操作數(shù)棧中,并且把PC的值指向 方法調(diào)用指令 后面
的一條指令地址。
c. 操作數(shù)棧:操作變量的內(nèi)存模型。操作數(shù)棧的最大深度在編譯的時(shí)候已經(jīng)確定(寫入方法區(qū)code屬性的
max_stacks項(xiàng)中)。操作數(shù)棧的的元素可以是任意Java類型,包括long和double,32位數(shù)據(jù)占用??臻g
為1,64位數(shù)據(jù)占用2。方法剛開始執(zhí)行的時(shí)候,棧是空的,當(dāng)方法執(zhí)行過程中,各種字節(jié)碼指令往棧中
存取數(shù)據(jù)。
d. 動(dòng)態(tài)鏈接:每個(gè)棧幀都持有在運(yùn)行時(shí)常量池中該棧幀所屬方法的引用,持有這個(gè)引用是為了支持方法調(diào)
用過程中的動(dòng)態(tài)鏈接。
3.本地方法棧(線程私有):調(diào)用本地native的內(nèi)存模型
4.方法區(qū)(線程共享):用于存儲(chǔ)已被虛擬機(jī)加載的類信息、常量、靜態(tài)變量,即時(shí)編譯后的代碼等數(shù)據(jù)。
運(yùn)行時(shí)常量池:方法區(qū)的一部分,存放編譯期生成的各種字面量和符號(hào)引用;Class文件中除了存有類的版本、字段、方法、接口等描述信息,還有一項(xiàng)是常量池,存有這個(gè)類的 編譯期生成的各種字面量和符號(hào)引用,這部分內(nèi)容將在類加載后,存放到方法區(qū)的運(yùn)行時(shí)常量池中。
5.堆區(qū) (線程共享):Java對(duì)象存儲(chǔ)的地方
a. Java堆是虛擬機(jī)管理內(nèi)存最大的一塊
b. 存放對(duì)象實(shí)例,幾乎所有的對(duì)象實(shí)例都在這里分配內(nèi)存,存放new生成對(duì)象和數(shù)組
c. Java堆是垃圾收集器管理的內(nèi)存區(qū)域,因此很多時(shí)候稱為“GC堆”