除了程式計數器外,虛擬機記憶體在其他幾個運行時區域都有發生OutOfMemoryError異常的可能。 Java堆溢出 設置Idea堆的大小為20MB,不可擴展(-Xms參數與最大值-Xmx參數設置為一樣,避免自動擴展) -verbose:gc -Xms20M -Xmx20M -Xmn10M -XX: ...
除了程式計數器外,虛擬機記憶體在其他幾個運行時區域都有發生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)