OutOfMemoryError異常

来源:https://www.cnblogs.com/xtt321/archive/2022/05/03/16218167.html
-Advertisement-
Play Games

除了程式計數器外,虛擬機記憶體在其他幾個運行時區域都有發生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)

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • pandas讀取Excel、csv文件中的數據時,得到的大多是表格型的二維數據,在pandas中對應的即為DataFrame數據結構。在處理這類數據時,往往要根據據需求先獲取數據中的子集,如某些列、某些行、行列交叉的部分等。可以說子集選取是一個非常基礎、頻繁使用的操作,而DataFrame的子集選取 ...
  • 8. 文件讀寫操作 1 #include<iostream> 2 #include<string> 3 #include<fstream> // 讀寫文件 頭文件 4 using namespace std; 5 6 // 文件操作 7 8 // 寫文件 9 void writefile() { 1 ...
  • 以下是我收集的一些問題,有的是網上摘錄的,有的是自己參加面試被問到的,有的是工作或學習時遇到的,等等。 為什麼要記錄這些呢? 一方面,我相信,這樣做對我自己的技術提升是有幫助的。在全文結構上我儘量**使問題連貫地形成知識體系**,而不是堆積的碎片,而且,每個問題我會儘量地給出答案。 另一方面,我希望... ...
  • VSCode開發環境配置 先到VSCode官網去下載適合自己系統的VSCode安裝軟體 VScode下載地址:https://code.visualstudio.com/Download ### 演示在WIndows下 安裝使用 (1)把vscode安裝軟體準備好 如果不清楚選64位還是32位可以在 ...
  • 文件操作(輸入輸出流) 文件操作的概述 程式運行時產生的數據都屬於零食數據,程式一旦運行結束,就會被釋放 通過文件可以將數據持久化 C++中對文件的操作包含頭文件(文件流) 文件類型分為兩種 文本文件:文件以文本的ASCII碼的形式存儲在電腦中 二進位文件:文件以文本的二進位形式存儲在電腦中,用 ...
  • string是C標準模板庫中專門用於字元串處理的數據結構類型。它並不是 C的基本數據類型,它是 C++標準模板庫中的一個“類”。若要使用 string 對象,則必須包含頭文件#include <string>。 初始化 常用的初始化有以下幾種,帶等號的是拷貝初始化, string str1("hel ...
  • 7. 多態 7.1 多態基本用法 1 #include<iostream> 2 using namespace std; 3 4 // 多態 5 6 // 動態多態滿足條件: 7 // 1.有繼承關係 8 // 2. 子類重寫父類的虛函數 9 // 10 // 動態多態使用 11 // 父類的指針或 ...
  • 1.安裝 1.1 創建虛擬環境 mkdir myproject cd myproject python3 -m venv venv 1.2 進入虛擬環境 . venv/bin/activate 1.3 安裝 flask pip install Flask 2.上手 2.1 最小 Demo 將下列代碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...