0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

一文讀懂JVM是什么

冬至配餃子 ? 來源:極術社區(qū) ? 作者:betterFighter ? 2022-07-28 15:36 ? 次閱讀

前言

由于先前也遇到過一些性能問題,OOM算是其中的一大類了。因此也對jvm產(chǎn)生了一些興趣。自己對jvm略做了些研究。后續(xù)繼續(xù)補充。

從oom引申出去

既然說到oom,首先需要知道oom的原因是什么。為啥會oom嘞?
oom的定義是outofmemory。當內(nèi)存想為對象分配內(nèi)存的時候,發(fā)現(xiàn)內(nèi)存不足以去分配內(nèi)存,或者gc的時候發(fā)現(xiàn)沒有可以被回收的對象或回收后的內(nèi)存也不足以為對象分配內(nèi)存。

因此拋出這個java異常。

oom
可以分為以下四類

1.堆溢出:java堆

2.棧溢出:虛擬機棧和本地方法棧

3.方法區(qū)內(nèi)存溢出:方法區(qū)和內(nèi)存時常量池

4.本機直接內(nèi)存溢出

因此,需要先了解堆,棧,方法區(qū)都是些啥

運行時數(shù)據(jù)區(qū)

先上圖

poYBAGLiO6-AYRh8AABaoIqp-Dk913.png

程序計數(shù)器:當前線程所執(zhí)行的字節(jié)碼的行號指示器。

java虛擬機的多線程是通過輪流切換線程,并為線程分配執(zhí)行時間片去運行來執(zhí)行的。每個線程都有一個自己的程序計數(shù)器。我覺得這個可以這么理解:當一個線程在運行的時候,每執(zhí)行一步程序計數(shù)器都會有個記錄,記錄當前執(zhí)行到哪一步了。如果線程被切換后又切換回來,那么通過程序計數(shù)器就能知道執(zhí)行到哪一步了,然后繼續(xù)向下執(zhí)行。

虛擬機棧:每個線程都會有一個虛擬機棧。虛擬機棧描述的是java方法執(zhí)行的內(nèi)存模型。因為線程執(zhí)行的過程就是執(zhí)行線程里的一個個方法,而每個方法都會創(chuàng)建對應自己的棧幀。

棧幀里存的內(nèi)容如下:

局部變量表:存放了各種編譯期可知基本數(shù)據(jù)類型,對象引用(引用指針或句柄)

操作數(shù)棧:大多數(shù)指令都要從這里彈出數(shù)據(jù),執(zhí)行運算,然后把結果壓回操作數(shù)棧

動態(tài)鏈接

方法出口

64位的long和都double類型數(shù)據(jù)占用2個局部變量空間,其他數(shù)據(jù)類型占用一個,也就是每個局部變量空間為32位。

在這個地方,如果線程請求的深度大于虛擬機允許的深度,會拋出StackOverflowError.因為jvm分配給虛擬機棧的內(nèi)存是有限的,而每個方法都會有對應的棧幀壓入到棧中,如果調(diào)用方法過多,那么棧滿了自然也就溢出了。(可能的場景:死循環(huán)代碼,大量遞歸調(diào)用,那排查問題的時候也可以由此有一個思路)。可以通過調(diào)整-Xss去調(diào)整棧大小。

大部分java虛擬機允許動態(tài)擴展,但如果擴展的時候也申請不到足夠內(nèi)存時,就會報OOM了。

本地方法棧:和虛擬機發(fā)揮作用相似。區(qū)別:虛擬機棧為虛擬機執(zhí)行java方法服務,本地方法棧為虛擬機使用的Native方法服務。Native Method就是一個java調(diào)用非java代碼的接口,Native方法的實現(xiàn)由非java語言實現(xiàn)。讀者不用糾結,略作了解即可。

:堆是所有線程共享的一塊內(nèi)存,作用是存放對象實例。堆可以分為新生代和老年代。新生代里還可細分為Eden,From survivor,To survivor等空間。后面講述GC過程時會說到。

方法區(qū):也是所有線程共享的一塊內(nèi)存,存放被虛擬機加載的類信息,常量,靜態(tài)變量,編譯器編譯后的代碼。也就是常說的永久代。

永久代的大小可以用-XX:MaxPermSize去設置。

運行時常量池:方法區(qū)的一部分。存放編譯期生成的各種字面量和符號引用。字面量就是指這個量本身。比如字面量2,就是指2.

運行時常量池有一個重要特性就是動態(tài)性。常量不一定只有編譯期才能產(chǎn)生,運行期間也可能將新的常量放入常量池。詳情可見String類的

intern()方法。

直接內(nèi)存:它不是虛擬機運行時數(shù)據(jù)區(qū)的一部分,但也頻繁的被使用。直接內(nèi)存不會受到java堆大小的限制,但是會受到本機總內(nèi)存的限制。

GC過程

GC分為新生代GC(minor gc)和老年代GC(full gc)。新生代GC的頻率遠遠高于老年代。而且

新生代GC的速度會比老年代的GC速度快10倍以上。根源在于新生代和老年代使用的GC算法不同。讀者們可以去仔細思考下(答案文中有,哈哈)。新生代/老年代大小默認為1:2。

新生代GC過程

新生代里可細分為Eden,From survivor,To survivor等空間。當我們需要給對象分配內(nèi)存的時候,首先我們會在Eden區(qū)為對象分配內(nèi)存,當Eden區(qū)內(nèi)存不足時,會發(fā)生minor gc,此時會把仍然存活的對象放到From survivor,并給對象標記存活次數(shù)1;然后當Eden區(qū)再次被用完后,對Eden區(qū)和From survivor區(qū)篩選出存活的對象,放到To survivor區(qū),清空Eden區(qū)和From survivor區(qū),存活次數(shù)加1,之前存活的就是2了。

以此類推,默認是當存活次數(shù)到達15次(可配置)的時候,把這個對象存入老年代中。同時也可以看到,F(xiàn)rom survivor,To survivor區(qū)始終有一個是空置的。所以新生代使用的只有9/10的空間。
然而大家可以思考一下。Eden區(qū)和survivor區(qū)的大小為8:1,那么發(fā)生minor gc后如果存活的對象
的大小比survivor區(qū)還要大。這個時候會怎么處理?

這里需要引入一個叫“內(nèi)存分配擔保機制”的概念。就是當存活的對象連survivor區(qū)都放不下的時候,這部分放不下的對象會直接進入老年代。老年代是擔保人。老年代進行擔保,前提是老年代還有剩余空間。但是每次存活下來的對象大小是不確定的。所以只好取之前每次存儲到老年代的對象大小的平均值。如果大于平均值,那么直接full gc。但是為了避免頻繁full gc,仍然會開啟handlepromotionfailure配置。如下圖

poYBAGLiO9SAQQccAAAYbuaIYlY922.png

老年代GC過程

老年代采用了標記整理,標記清楚的算法。老年代會把仍然存活的對象都整理統(tǒng)一放到一邊。整理完成后就會清楚掉邊界外的對象。這樣就避免了產(chǎn)生大量的內(nèi)存碎片的問題。但是整理算法相較于新生代采用的復制算法,復雜程度肯定更高。這也導致了full gc的速度要遠遠慢于minor gc。


審核編輯:劉清

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2970

    瀏覽量

    104814
  • 計數(shù)器
    +關注

    關注

    32

    文章

    2256

    瀏覽量

    94646
  • 虛擬機
    +關注

    關注

    1

    文章

    918

    瀏覽量

    28232
  • 內(nèi)存溢出

    關注

    0

    文章

    10

    瀏覽量

    1229
收藏 人收藏

    評論

    相關推薦

    讀懂單燈控制器工作原理

    讀懂單燈控制器工作原理
    的頭像 發(fā)表于 11-11 13:13 ?268次閱讀
    <b class='flag-5'>一</b><b class='flag-5'>文</b><b class='flag-5'>讀懂</b>單燈控制器工作原理

    讀懂MSA(測量系統(tǒng)分析)

    讀懂MSA(測量系統(tǒng)分析)
    的頭像 發(fā)表于 11-01 11:08 ?955次閱讀
    <b class='flag-5'>一</b><b class='flag-5'>文</b><b class='flag-5'>讀懂</b>MSA(測量系統(tǒng)分析)

    讀懂新能源汽車的功能安全

    電子發(fā)燒友網(wǎng)站提供《讀懂新能源汽車的功能安全.pdf》資料免費下載
    發(fā)表于 09-04 09:22 ?3次下載

    從原理聊JVM):染色標記和垃圾回收算法

    導讀 JAVA簡單易用的特性,能夠讓研發(fā)人員在不了解JVM的底層運行機制的情況下依舊能夠編寫出功能完善的代碼。 但是對JVM的理解,是個程序員普通和優(yōu)秀的分水嶺。全面地了解JVM的工
    的頭像 發(fā)表于 08-20 15:25 ?247次閱讀
    從原理聊<b class='flag-5'>JVM</b>(<b class='flag-5'>一</b>):染色標記和垃圾回收算法

    聊聊JVM如何優(yōu)化

    首先應該明確的是JVM調(diào)優(yōu)不是常規(guī)手段,JVM的存在本身就是為了減輕開發(fā)對于內(nèi)存管理的負擔,當出現(xiàn)性能問題的時候第時間考慮的是代碼邏輯與設計方案,以及是否達到依賴中間件的瓶頸,最后才是針對J
    的頭像 發(fā)表于 08-05 17:49 ?486次閱讀
    聊聊<b class='flag-5'>JVM</b>如何優(yōu)化

    【古瑞瓦特光伏逆變器品牌】讀懂PCS儲能變流器

    【古瑞瓦特光伏逆變器品牌】讀懂PCS儲能變流器 在加快實現(xiàn)雙碳目標和構建新型電力系統(tǒng)的進程中,儲能技術正逐步成為支撐新型電力系統(tǒng)穩(wěn)定運行、優(yōu)化資源配置的關鍵技術之。其中,PCS(
    的頭像 發(fā)表于 06-14 16:39 ?1333次閱讀
    【古瑞瓦特光伏逆變器品牌】<b class='flag-5'>一</b><b class='flag-5'>文</b><b class='flag-5'>讀懂</b>PCS儲能變流器

    讀懂泰凌微電子2023年年度報告

    讀懂泰凌微電子2023年年度報告
    的頭像 發(fā)表于 05-22 10:30 ?469次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>泰凌微電子2023年年度報告

    讀懂芯導科技2024年第季度報告

    讀懂芯導科技2024年第季度報告
    的頭像 發(fā)表于 04-26 11:31 ?571次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>芯導科技2024年第<b class='flag-5'>一</b>季度報告

    讀懂廣立微2023年度報告

    讀懂廣立微2023年度報告
    的頭像 發(fā)表于 04-22 10:00 ?391次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>廣立微2023年度報告

    讀懂億緯鋰能2023年度報告

    讀懂億緯鋰能2023年度報告
    的頭像 發(fā)表于 04-19 10:33 ?437次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>億緯鋰能2023年度報告

    讀懂紫光國微2023年報

    志高行遠 萬里可期 | 讀懂紫光國微2023年報
    的頭像 發(fā)表于 04-18 10:40 ?537次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>紫光國微2023年報

    讀懂芯導科技2023年年度報告

    讀懂芯導科技2023年年度報告
    的頭像 發(fā)表于 04-16 14:18 ?379次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>芯導科技2023年年度報告

    讀懂概倫電子2023年年報

    踔厲奮發(fā),向新而行 | 讀懂概倫電子2023年年報
    的頭像 發(fā)表于 04-14 16:10 ?618次閱讀
    <b class='flag-5'>一</b>圖<b class='flag-5'>讀懂</b>概倫電子2023年年報

    電主軸:教您如何讀懂?|深圳恒興隆機電.

    電主軸:教您如何讀懂?|深圳恒興隆機電電主軸是種利用電機作為驅動源的主軸。它具有高速、高精度和高剛性等特點,廣泛應用于機械加工、數(shù)控機床、機器人等領域。本文將詳細介紹電主軸的原理
    發(fā)表于 03-27 10:30

    讀懂寬帶、帶寬、網(wǎng)速之間的區(qū)別與關系

    讀懂寬帶、帶寬、網(wǎng)速之間的區(qū)別與關系? 寬帶、帶寬和網(wǎng)速是在網(wǎng)絡領域中經(jīng)常使用的術語,它們之間有定的區(qū)別和關系。在深入理解寬帶、帶寬和網(wǎng)速之間的關系之前,讓我們先了解
    的頭像 發(fā)表于 01-31 09:11 ?7293次閱讀