除了程式計數器外,虛擬機記憶體在其他幾個運行時區域都有發生OutOfMemoryError例外的可能,
Java堆溢位
設定Idea堆的大小為20MB,不可擴展(-Xms引數與最大值-Xmx引數設定為一樣,避免自動擴展)

-verbose:gc -Xms20M -Xmx20M -Xmn10M -XX:+PrintGCDetails -XX:SurvivorRatio=8
運行以下代碼:
package memory; import java.util.ArrayList; import java.util.List; public class HeepOOM { static class OOMObject{ } public static void main(String[] args){ List<OOMObject> list = new ArrayList<>(); while (true){ list.add(new OOMObject()); } } } 拋出錯誤: Exception in thread "main" java.lang.OutOfMemoryError: Java heap space at java.base/java.util.Arrays.copyOf(Arrays.java:3512) at java.base/java.util.Arrays.copyOf(Arrays.java:3481) at java.base/java.util.ArrayList.grow(ArrayList.java:237) at java.base/java.util.ArrayList.grow(ArrayList.java:244) at java.base/java.util.ArrayList.add(ArrayList.java:454) at java.base/java.util.ArrayList.add(ArrayList.java:467) at memory.HeepOOM.main(HeepOOM.java:14)
解決這個例外重點是確認記憶體中的物件是否是必要的,也就是區分是出現了記憶體泄漏(Memory Leak)還是記憶體溢位(Memory Overflow)
- 記憶體泄漏:程式申請記憶體后,無法釋放已申請的記憶體空間,多次記憶體泄漏堆積后的后果就是記憶體溢位
- 記憶體溢位:程式申請記憶體時,沒有足夠的記憶體供申請者使用
如果是記憶體泄漏,可進一步通過工具查看泄漏物件到GC Roots的參考鏈,于是就能找到泄漏物件是通過怎樣的路徑與GC Root相關聯導致垃圾回收器無法自動回收他們,
如果不存在泄漏,則就應該檢查虛擬機堆引數(-Xmx與-Xms),與機器物理記憶體對比看是否還可以調大,從代碼上檢查是否存在某些物件生命周期過長、持有狀態時間過長的情況,嘗試減少長期運行期的記憶體消耗,
虛擬機堆疊和本地方法堆疊溢位
- 如果執行緒請求的堆疊深度大于虛擬機所允許的最大深度,則拋出StackOverflowError例外,
- 如果虛擬機在擴展堆疊時無法申請到足夠的記憶體,則拋出OutOfMemoryError例外,
package memory; public class JavaVmStackSOF { private int stackLength = 1; public void stackLeak(){ stackLength++; stackLeak(); } /** * -Xss180K -設定每個執行緒分配180K記憶體 * @param args * @throws Throwable */ public static void main(String[] args) throws Throwable{ JavaVmStackSOF oom = new JavaVmStackSOF(); try{ oom.stackLeak(); }catch (Throwable e){ System.out.println("stack length:"+oom.stackLength); throw e; } } }
運行結果:
Exception in thread "main" java.lang.StackOverflowError stack length:1554 at memory.JavaVmStackSOF.stackLeak(JavaVmStackSOF.java:7)
package memory; public class JavaVMStackOOM { private void dontStop(){ while (true){ } } public void stackLeakByThread(){ while (true){ Thread thread = new Thread(new Runnable() { @Override public void run() { dontStop(); } }); thread.start(); } } /** * -Xss2M -設定每個執行緒分配2M記憶體 * 最侄訓耗盡所有記憶體,導致沒有足夠的記憶體創建執行緒而拋出例外:OutOfMemoryError * @param args * @throws Throwable */ public static void main(String[] args) throws Throwable{ JavaVMStackOOM oom = new JavaVMStackOOM(); oom.stackLeakByThread(); } }
運行結果
Exceptuib in thread "main" java.lang.OutOfMemoryError: unable to create new native thread
運行時常量池溢位
運行時常量池屬于方法區,因此可以通過以下方式模擬:
java7可以通過設定:-XX:PermSize=10M -XX:MaxPermSize=10M 來限定方法區,
java8之后永久代被移除,-XX:PermSize、-XX:MaxPermSize已經被移除;可以使用:-XX:MetaspaceSize=10M -XX:MaxMetaspaceSize=10M (元空間代替)
型別資訊、欄位、方法、常量保存在元空間,
package main.java.loadclass; import java.util.ArrayList; import java.util.List; public class RuntimeConstantPoolOOM { /** * JAVA7 下運行 * @param args */ public static void main(String[] args){ //使用LIST保持著常量池參考,避免Full GC 回收常量池行為 List<String> list = new ArrayList<>(); //10MB的PermSize在integer范圍內足夠產生OOM了 int i = 0; while (true){ list.add(String.valueOf(i++).intern()); } } }
拋出以下例外:java.lang.OutOfMemoryError:PermGen space
方法區溢位
方法區用于存放Class的相關資訊,如類名、訪問修飾符、常量池、欄位描述、方法描述等,一個類如果被垃圾回收器回收掉,判定條件非常苛刻,在經常動態生成大量Class的應用中,需要特別注意類的回收狀態,
本機直接記憶體溢位
DirectMemory容量可通過-XX:MaxDirectMemorySize指定,如果不指定,則默認與Java堆的最大值(-Xmx指定)一樣,
DirectByteBuffer是Java用于實作堆外記憶體的一個重要類,我們可以通過該類實作堆外記憶體的創建、使用和銷毀,DirectByteBuffer跑出記憶體溢位例外時并沒有真正整整向作業系統申請分配記憶體,而是通過計算得知記憶體無法分配,手動跑出例外,
使用Unsafe類的allocateMemory方法是真正分配記憶體,
package main.java.loadclass; import sun.misc.Unsafe; import java.lang.reflect.Field; public class DirectMemoryOOM { private static final int _1MB = 1024 * 1024; /** * -Xmx20M -XX:MaxDirectMemorySize=10M * @param args * @throws Exception */ public static void main(String[] args) throws Exception{ Field unsefeField = Unsafe.class.getDeclaredFields()[0]; unsefeField.setAccessible(true); Unsafe unsafe = (Unsafe) unsefeField.get(null); while (true){ unsafe.allocateMemory(_1MB); } } }
回傳結果:
Exception in thread "main" java.lang.OutOfMemoryError at java.base/jdk.internal.misc.Unsafe.allocateMemory(Unsafe.java:616) at jdk.unsupported/sun.misc.Unsafe.allocateMemory(Unsafe.java:462) at main.java.loadclass.DirectMemoryOOM.main(DirectMemoryOOM.java:16)

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/469590.html
標籤:其他
上一篇:Python 函式進階-遞回函式
