JVM運(yùn)行時(shí)數(shù)據(jù)模型

深入理解JAVA虛擬機(jī)中說(shuō)道的一句話,感覺(jué)很符合Java和C++程序員在處理內(nèi)存時(shí)的心態(tài):

Java和C++之間有一堵由內(nèi)存動(dòng)態(tài)分配和垃圾回收技術(shù)所圍成的“高墻”,墻外的人想進(jìn)去循帐,墻里的人想出去

Java運(yùn)行時(shí)數(shù)據(jù)模型

Java運(yùn)行時(shí)數(shù)據(jù)模型

程序計(jì)數(shù)器(Program Counter Register)

? 程序計(jì)數(shù)器是一塊較小的內(nèi)存空間,它可以看做是當(dāng)前線程所執(zhí)行的字節(jié)碼的行號(hào)指示器。字節(jié)碼解釋器工作時(shí)瑰钮,就是通過(guò)改變這個(gè)計(jì)數(shù)器的值來(lái)選取下一條需要執(zhí)行的字節(jié)碼指令。

? Java虛擬機(jī)的多線程是通過(guò)線程輪流切換并分配處理器執(zhí)行時(shí)間的方式來(lái)實(shí)現(xiàn)的微驶,在任何一個(gè)確認(rèn)的時(shí)刻浪谴,一個(gè)處理器(多核的話,一個(gè)內(nèi)核)都只會(huì)執(zhí)行一條線程中的指令因苹。因此苟耻,為了線程切換后,能恢復(fù)到正確的執(zhí)行位置扶檐,每條線程都需要有一個(gè)獨(dú)立的程序計(jì)數(shù)器凶杖。

? 如果正在執(zhí)行的是Java方法,則程序計(jì)數(shù)器指向的是下一條需要執(zhí)行的字節(jié)碼指令的地址款筑;如果正在執(zhí)行的是Native方法智蝠,則程序計(jì)數(shù)器為空(Undefined)腾么。

此區(qū)域是唯一一個(gè)在Java虛擬機(jī)規(guī)范中沒(méi)有規(guī)定任何OutOfMemoryError情況的區(qū)域

Java虛擬機(jī)棧

? Java虛擬機(jī)棧也是線程私有的,生命周期也是和線程掛鉤杈湾。

? Java虛擬機(jī)棧描述的是Java方法執(zhí)行的內(nèi)存模型解虱。每個(gè)Java方法在執(zhí)行的同時(shí),都會(huì)創(chuàng)建一個(gè)棧幀(Stack Frame)漆撞,用于存儲(chǔ)方法的:局部變量表殴泰,操作數(shù)棧動(dòng)態(tài)鏈接浮驳,方法出口等信息艰匙。 每個(gè)方法在從調(diào)用知道執(zhí)行結(jié)束的過(guò)程,就對(duì)應(yīng)著一個(gè)棧幀在虛擬機(jī)棧中的入棧到出棧的過(guò)程抹恳。

  1. 局部變量表:局部變量表存放了編譯器可知的各種基本數(shù)據(jù)類型(boolean, byte,char,short, int , float, long, double),對(duì)象引用(Reference類型)和returnAddress類型(指向了一條字節(jié)碼指令的地址)员凝。

    1. 局部變量空間(slot):在32位系統(tǒng)中,它的大小一般是4個(gè)byte奋献,也就是32位健霹。

    只有double和long類型占用2個(gè)局部變量空間,其他數(shù)據(jù)類型都是占用一個(gè)slot瓶蚂,在進(jìn)入一個(gè)方法時(shí)糖埋,這個(gè)方法所在的棧幀需要分配多少局部變量空間是確定的,在方法運(yùn)行期間是不會(huì)改變的窃这。

  2. 操作數(shù)棧:

  3. 動(dòng)態(tài)鏈接

  4. 方法出口

在Java虛擬機(jī)規(guī)范中瞳别,對(duì)于虛擬機(jī)棧規(guī)定了兩種異常情況:

  1. 如果線程申請(qǐng)的棧深度大于虛擬機(jī)所允許的深度,將拋出StackOverflowError異常杭攻。
  2. 如果虛擬機(jī)椝盍玻可以動(dòng)態(tài)擴(kuò)展,如果擴(kuò)展時(shí)無(wú)法申請(qǐng)到足夠內(nèi)存兆解,就會(huì)拋出OutOverMemoryError異常

本地方法棧

? 本地方法棧與虛擬機(jī)棧所發(fā)揮的作用非常類似馆铁,只不過(guò)虛擬機(jī)棧為虛擬機(jī)執(zhí)行Java方法(也就是字節(jié)碼)服務(wù),本地方法棧是為虛擬機(jī)使用到的Native方法服務(wù)锅睛。

? 在Java 虛擬機(jī)規(guī)范中埠巨,并沒(méi)有對(duì)Native方法中使用的語(yǔ)言,使用方式和數(shù)據(jù)結(jié)構(gòu)做強(qiáng)制規(guī)定现拒。所以具體的虛擬機(jī)可以自由的實(shí)現(xiàn)它辣垒,甚至有的虛擬機(jī)(Sun HotSpot虛擬)直接把本地方法棧與虛擬機(jī)棧合二為一。

在Java虛擬機(jī)規(guī)范中印蔬,對(duì)于本地方法棧也規(guī)定了兩種異常情況:

  1. 如果線程申請(qǐng)的棧深度大于虛擬機(jī)所允許的深度分冈,將拋出StackOverflowError異常戚嗅。
  2. 如果虛擬機(jī)椚悍觯可以動(dòng)態(tài)擴(kuò)展,如果擴(kuò)展時(shí)無(wú)法申請(qǐng)到足夠內(nèi)存岂丘,就會(huì)拋出OutOverMemoryError異常

Java堆(Java Heap)

? Java堆是虛擬管理的最大的一塊內(nèi)存空間,并且是線程共享的一塊內(nèi)存空間眠饮,堆在虛擬機(jī)啟動(dòng)時(shí)創(chuàng)建奥帘,此內(nèi)存空間唯一的目的就是存放對(duì)象實(shí)例。幾乎所有的對(duì)象實(shí)例都是在堆上分配內(nèi)存仪召。Java虛擬機(jī)規(guī)范中描述的是:所有的對(duì)象實(shí)例以及數(shù)組都要在堆上分配寨蹋。

? 但是隨著技術(shù)的發(fā)展,JIT即時(shí)編譯器的發(fā)展以及逃逸分析技術(shù)的逐漸成熟:棧上分配扔茅,標(biāo)量替換優(yōu)化技術(shù)將會(huì)導(dǎo)致一些微妙的變化已旧。所有對(duì)象在堆上進(jìn)行分配也逐漸的變得不是那么“絕對(duì)”了。

? 由于Java堆是垃圾收集器管理的主要區(qū)域召娜, 因此很多時(shí)候Java堆也叫GC堆运褪。

? 由于現(xiàn)在的垃圾回收器都是使用分代收集算法,所以Java堆還可以細(xì)分為:新生代和老年代玖瘸。在細(xì)致一些秸讹,可分為:Eden區(qū)域,TO Survivor區(qū)域和FROM Survivor區(qū)域雅倒。

? 根據(jù)Java虛擬機(jī)規(guī)范璃诀,Java堆可以處于物理上不連續(xù)的內(nèi)存空間,只要邏輯上是連續(xù)的即可∶锵唬現(xiàn)在的主流的虛擬機(jī)都是按照可擴(kuò)展來(lái)實(shí)現(xiàn)的(通過(guò)-Xmx和-Xms實(shí)現(xiàn))劣欢。

? 在Java虛擬機(jī)規(guī)范中,對(duì)于Java堆規(guī)定了一種異常情況:當(dāng)堆中沒(méi)有內(nèi)存完成對(duì)象實(shí)例裁良,并且無(wú)法在進(jìn)行擴(kuò)展時(shí)凿将,會(huì)拋出OutOfMemoryError異常。

方法區(qū)(Method Area)

? 方法區(qū)與Java堆一樣趴久,是各個(gè)線程共享的一塊內(nèi)存區(qū)域丸相,用于存儲(chǔ)已經(jīng)被虛擬機(jī)加載的類信息,常量彼棍,靜態(tài)變量,即時(shí)編譯器編譯后的代碼等數(shù)據(jù)膳算。

? 雖然Java虛擬機(jī)規(guī)范將方法區(qū)描述為堆的一個(gè)邏輯部分座硕,但是他還有另外一個(gè)名字:Non-Heap(非堆)

? Java虛擬機(jī)規(guī)范對(duì)方法區(qū)的限制非常寬泛,除了像Java堆一樣不需要連續(xù)的物理內(nèi)存涕蜂,可以選擇固定大小或者可擴(kuò)展外华匾,甚至還可以選擇不實(shí)現(xiàn)垃圾回收。垃圾回收行為在這個(gè)區(qū)域是比較少出現(xiàn)的,但這不意味著這個(gè)區(qū)域的垃圾回收是不必要的蜘拉,這塊區(qū)域進(jìn)行垃圾回收的目的主要是:1. 針對(duì)常量池的回收萨西; 2. 類型的卸載

在Sun公司的BUG列表中,曾出現(xiàn)過(guò)若干嚴(yán)重的BUG就是由于低版本的HotSpot虛擬機(jī)對(duì)此區(qū)域未完全回收而導(dǎo)致的內(nèi)存泄漏

? 在Java虛擬機(jī)規(guī)范中旭旭,對(duì)于方法區(qū)規(guī)定了一種異常情況:當(dāng)方法區(qū)中無(wú)法滿足內(nèi)存分配需求時(shí)谎脯,會(huì)拋出OutOfMemoryError異常。

方法區(qū) --> 運(yùn)行時(shí)常量池

? 運(yùn)行時(shí)常量池屬于方法區(qū)的一部分持寄。

? Class文件中除了有類的版本源梭,字段,方法稍味,接口等描述信息外废麻,還有一項(xiàng)是常量池(Constant Pool Table),用于存放編譯期生成的各種字面量符號(hào)引用模庐。這部分內(nèi)容會(huì)在類加載后進(jìn)入到方法區(qū)的運(yùn)行時(shí)常量池中存放烛愧。

? Java虛擬機(jī)規(guī)范并沒(méi)有對(duì)運(yùn)行時(shí)常量池做任何細(xì)節(jié)上的要求

? 運(yùn)行時(shí)常量池相對(duì)于Class文件中的常量池的另外一個(gè)重要特征就是動(dòng)態(tài)。Java語(yǔ)言并不要求常量一定只有編譯期才能產(chǎn)生掂碱,運(yùn)行時(shí)也可以將新的變量放入池中怜姿。比如String.intern()方法。

? 由于運(yùn)行時(shí)常量池屬于方法區(qū)顶吮,自然受到方法區(qū)內(nèi)存的限制社牲,當(dāng)運(yùn)行時(shí)常量池?zé)o法在申請(qǐng)到內(nèi)存時(shí),將拋出OutOfMemoryError異常悴了。

直接內(nèi)存(Dirct Memory)

? 直接內(nèi)存并不屬于Java內(nèi)存模型搏恤,不是JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)的一部分,它會(huì)被頻繁調(diào)用是因?yàn)樵?.4版本引入的NIO(New Input/Output)類湃交,引入了一種基于通道(Channel)和緩存(Buffer)的I/O方式熟空。

? 它可以使用Native函數(shù)庫(kù)直接分配堆外內(nèi)存,然后通過(guò)一個(gè)儲(chǔ)存在堆上的DirectByteBuffer對(duì)象作為這塊內(nèi)存的引用進(jìn)行操作搞莺。這樣做息罗,能顯著提高性能,因?yàn)楸苊饬嗽贘ava堆和Native堆中來(lái)回復(fù)制數(shù)據(jù)才沧。

? 顯然迈喉,直接內(nèi)存的分配不受Java堆大小的限制,但是會(huì)受本機(jī)總內(nèi)存(包括RAM以及SWAP區(qū)或者分頁(yè)文件)大小以及處理器尋址空間的限制温圆。

? 直接內(nèi)存出現(xiàn)OutOfMemoryError的原因是對(duì)該區(qū)域進(jìn)行內(nèi)存分配時(shí)挨摸,其內(nèi)存與其他內(nèi)存加起來(lái)超過(guò)最大物理內(nèi)存限制(包括物理的和操作系統(tǒng)級(jí)的限制),從而導(dǎo)致OutOfMemoryError岁歉。另外得运,若我們通過(guò)參數(shù)“-XX:MaxDirectMemorySize”指定了直接內(nèi)存的最大值,其超過(guò)指定的最大值時(shí),也會(huì)拋出內(nèi)存溢出異常熔掺。

總結(jié)

? 這篇文章主要是介紹Java虛擬機(jī)運(yùn)行時(shí)數(shù)據(jù)區(qū)中的分區(qū)饱搏,并介紹了每個(gè)區(qū)的概念,主要儲(chǔ)存什么數(shù)據(jù)置逻,異常情況推沸。

? 理解Java虛擬機(jī)運(yùn)行時(shí)數(shù)據(jù)區(qū),可以有效的幫助我們了解虛擬機(jī)是如果使用內(nèi)存的诽偷,對(duì)排查錯(cuò)誤坤学,快速定位問(wèn)題很有幫助。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末报慕,一起剝皮案震驚了整個(gè)濱河市深浮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌眠冈,老刑警劉巖飞苇,帶你破解...
    沈念sama閱讀 222,946評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異蜗顽,居然都是意外死亡布卡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,336評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)雇盖,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)忿等,“玉大人,你說(shuō)我怎么就攤上這事崔挖∶辰郑” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,716評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵狸相,是天一觀的道長(zhǎng)薛匪。 經(jīng)常有香客問(wèn)我,道長(zhǎng)脓鹃,這世上最難降的妖魔是什么逸尖? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,222評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮瘸右,結(jié)果婚禮上娇跟,老公的妹妹穿的比我還像新娘。我一直安慰自己太颤,他們只是感情好逞频,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,223評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著栋齿,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上瓦堵,一...
    開(kāi)封第一講書(shū)人閱讀 52,807評(píng)論 1 314
  • 那天基协,我揣著相機(jī)與錄音,去河邊找鬼菇用。 笑死澜驮,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的惋鸥。 我是一名探鬼主播杂穷,決...
    沈念sama閱讀 41,235評(píng)論 3 424
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼卦绣!你這毒婦竟也來(lái)了耐量?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 40,189評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤滤港,失蹤者是張志新(化名)和其女友劉穎廊蜒,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體溅漾,經(jīng)...
    沈念sama閱讀 46,712評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡山叮,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,775評(píng)論 3 343
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了添履。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片屁倔。...
    茶點(diǎn)故事閱讀 40,926評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖暮胧,靈堂內(nèi)的尸體忽然破棺而出锐借,到底是詐尸還是另有隱情,我是刑警寧澤叔壤,帶...
    沈念sama閱讀 36,580評(píng)論 5 351
  • 正文 年R本政府宣布瞎饲,位于F島的核電站,受9級(jí)特大地震影響炼绘,放射性物質(zhì)發(fā)生泄漏嗅战。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,259評(píng)論 3 336
  • 文/蒙蒙 一俺亮、第九天 我趴在偏房一處隱蔽的房頂上張望驮捍。 院中可真熱鬧,春花似錦脚曾、人聲如沸东且。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,750評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)珊泳。三九已至鲁冯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間色查,已是汗流浹背薯演。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,867評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留秧了,地道東北人跨扮。 一個(gè)月前我還...
    沈念sama閱讀 49,368評(píng)論 3 379
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像验毡,于是被迫代替她去往敵國(guó)和親衡创。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,930評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容