堆內(nèi)存中主要存放對象、數(shù)組等,只要不斷地創(chuàng)建這些對象,并且保證 GC Roots 到對象之間有可達路徑來避免垃 圾收集回收機制清除這些對象,當這些對象所占空間超過最大堆容量時,就會產(chǎn)生 OutOfMemoryError 的異常。堆 內(nèi)存異常示例如下:
運行后會報異常,在堆棧信息中可以看到
java.lang.OutOfMemoryError: Java heap space 的信息,說明在堆內(nèi)存空間產(chǎn)生內(nèi)存溢出的異常。
新產(chǎn)生的對象最初分配在新生代,新生代滿后會進行一次 Minor GC ,如果 Minor GC 后空間不足會把該對象和 新生代滿足條件的對象放入老年代,老年代空間不足時會進行 Full GC ,之后如果空間還不足以存放新對象則拋 出 OutOfMemoryError 異常。
常見原因:
由于HotSpot虛擬機中并不區(qū)分虛擬機棧和本地方法棧, 因此對于HotSpot來說, -Xoss參數(shù)(設(shè)置本地方法棧大 小) 雖然存在, 但實際上是沒有任何效果的, 棧容量只能由-Xss參數(shù)來設(shè)定。 關(guān)于虛擬機棧和本地方法棧, 在 《Java虛擬機規(guī)范》 中描述了兩種異常:
1) 如果線程請求的棧深度大于虛擬機所允許的最大深度, 將拋出StackOverflowError異常。
2) 如果虛擬機的棧內(nèi)存允許動態(tài)擴展, 當擴展棧容量無法申請到足夠的內(nèi)存時, 將拋出 OutOfMemoryError異 常。
《Java虛擬機規(guī)范》 明確允許Java虛擬機實現(xiàn)自行選擇是否支持棧的動態(tài)擴展, 而HotSpot虛擬機的選擇是不支持 擴展, 所以除非在創(chuàng)建線程申請內(nèi)存時就因無法獲得足夠內(nèi)存而出現(xiàn) OutOfMemoryError異常, 否則在線程運行時 是不會因為擴展而導致內(nèi)存溢出的, 只會因為棧容量無法容納新的棧幀而導致StackOverflowError異常。
為了驗證 這點, 我們可以做兩個實驗, 先將實驗范圍限制在單線程中操作, 嘗試下面兩種行為是 否能讓HotSpot虛擬機產(chǎn) 生OutOfMemoryError異常: 使用-Xss參數(shù)減少棧內(nèi)存容量。 結(jié)果: 拋出StackOverflowError異常, 異常出現(xiàn)時輸出 的堆棧深度相應(yīng)縮小。 定義了大量的本地變量, 增大此方法幀中本地變量表的長度。 結(jié)果: 拋出 StackOverflowError異常, 異常出現(xiàn)時輸出的堆棧深度相應(yīng)縮小。
由于運行時常量池是方法區(qū)的一部分, 所以這兩個區(qū)域的溢出測試可以放到一起進行。前面曾經(jīng)提到HotSpot從 JDK 7開始逐步“去永久代”的計劃, 并在JDK 8中完全使用元空間來代替永久代的背景故事, 在此我們就以測試代碼 來觀察一下, 使用“永久代”還是“元空間”來實現(xiàn)方法區(qū), 對程序有什么 實際的影響。
String::intern()是一個本地方法, 它的作用是如果字符串常量池中已經(jīng)包含一個等于此String對象的 字符串, 則返 回代表池中這個字符串的String對象的引用; 否則, 會將此String對象包含的字符串添加到常量池中, 并且返回此 String對象的引用。 在JDK 6或更早之前的HotSpot虛擬機中, 常量池都是分配在永久代中, 我們可以通過-XX: PermSize和-XX: MaxPermSize限制永久代的大小, 即可間接限制其中常量池的容量。
方法區(qū)的其他部分的內(nèi)容, 方法區(qū)的主要職責是用于存放類型的相關(guān)信息, 如類名、 訪問修飾符、 常量池、 字段 描述、 方法描述等。 對于這部分區(qū)域的測試, 基本的思路是運行時產(chǎn)生大量的類去填滿方法區(qū), 直到溢出為止。
直接內(nèi)存(Direct Memory) 的容量大小可通過-XX: MaxDirectMemorySize參數(shù)來指定, 如果不去指定, 則默認與 Java堆最大值(由-Xmx指定) 一致, 越過了DirectByteBuer類直接通 過反射獲取Unsafe實例進行內(nèi)存分配 (Unsafe類的getUnsafe()方法指定只有引導類加載器才會返回實例, 體現(xiàn)了設(shè)計者希望只有虛擬機標準類庫里面的 類才能使用Unsafe的功能,在JDK 10時才將Unsafe 的部分功能通過VarHandle開放給外部使用) ,
因為雖然使用 DirectByteBuer分配內(nèi)存也會拋出內(nèi)存溢出異常, 但它拋出異常時并沒有真正向操作系統(tǒng)申請分配內(nèi)存, 而是通 過計算得知內(nèi)存無法分配就會 在代碼里手動拋出溢出異常, 真正申請分配內(nèi)存的方法是Unsafe::allocateMemory()。
本文鏈接:http://www.www897cc.com/showinfo-26-151-0.htmlJVM優(yōu)化:實戰(zhàn)OutOfMemoryError異常
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com
上一篇: 一文搞定Java NIO,以及各種奇葩流