談談對 OOM 的認識

🍀 java.lang.StackOverflowError

package com.brian.interview.study.jvm.oom;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 12:50
 */
public class StackOverflowErrorDemo {
    public static void main(String[] args) {
        stackOverflowError();
    }

    private static void stackOverflowError() {
        stackOverflowError();  // Exception in thread "main" java.lang.StackOverflowError
    }
}

🍀 java.lang.OutOfMemoryError: Java heap space

package com.brian.interview.study.jvm.oom;

import java.util.Random;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 12:56
 */

public class JavaHeapSpaceDemo {
    public static void main(String[] args) {

//        byte[] bytes = new byte[80 * 1024 * 1024];

        String str = "hello";

        while (true) {
            str += str + new Random().nextInt(11111111) + new Random().nextInt(22222222);
            str.intern();
        }



        // Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
    }
}

🍀 java.lang.OutOfMemoryError: GC overhead limit exceeded

JVM參數配置演示
-Xms10m -Xmx10m -XX:MaxDirectMemorySize=5m

GC回收時間過長時會拋出 OutOfMemoryError。過長的定義是, 超過98%的時間用來做GC並且回收了不到2%的對內存
連續多次 GC 都只回收了不到2%的極端情況下才會拋出。假如不拋出 GC overhead limit 錯誤會發生什麼情況呢?
那就是 GC 清理的這麼點內存很快會再次填滿, 迫使 GC 再次執行。這樣就形成惡性循環,
CPU 使用率一直是 100%, 而 GC 卻沒有任何成果。
在這裏插入圖片描述

package com.brian.interview.study.jvm.oom;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 13:52
 */

import java.util.ArrayList;
import java.util.List;

/**
 * JVM參數配置演示
 * -Xms10m -Xmx10m -XX:+PrintGCDetails -XX:MaxDirectMemorySize=5m
 *
 * GC回收時間過長時會拋出 OutOfMemoryError。過長的定義是, 超過98%的時間用來做GC並且回收了不到2%的對內存
 * 連續多次 GC 都只回收了不到2%的極端情況下才會拋出。假如不拋出 GC overhead limit 錯誤會發生什麼情況呢?
 * 那就是 GC 清理的這麼點內存很快會再次填滿, 迫使 GC 再次執行。這樣就形成惡性循環,
 * CPU 使用率一直是 100%, 而 GC 卻沒有任何成果。
 */
public class GCOverheadDemo {
    public static void main(String[] args) {
        int i = 0;
        List<String> list = new ArrayList<>();

        try {
            while (true) {
                list.add(String.valueOf(++i).intern());
            }
        } catch (Throwable e) {
            System.out.println("******************i:" + i);
            e.printStackTrace();  // Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceeded
            throw e;
        }
    }
}

🍀 java.lang.OutOfMemoryError: Direct buffer memory

package com.brian.interview.study.jvm.oom;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 14:06
 */

import java.nio.ByteBuffer;

/**
 * 配置參數:
 * -Xms10m -Xmx10m -XX:+PrintGCDetails -XX:MaxDirectMemorySize=5m
 *
 * 故障現象:
 * Exception in thread "main" java.lang.OutOfMemoryError: Direct buffer memory
 *
 * 導致原因:
 *
 * 寫NIO程序經常使用 ByteBuffer 來讀取或者寫入數據, 這是一種基於通道(Channel)與緩衝區(Buffer)的I/O方式,
 * 它可以使用 Native 函數庫直接分配堆外內存, 然後通過一個存儲在 Java 堆裏面的 DirectByteBuffer 對象作爲這塊內存的引用進行操作。
 * 這樣能在一些場景中顯著提高性能, 因爲避免了在 Java 堆和 Native 堆中來回複製數據。
 *
 * ByteBuffer.allocate(capability) 第一種方式是分配 JVM 堆內存, 屬於 GC 管轄範圍, 由於需要拷貝所以速度相對較慢
 *
 * ByteBuffer.allocateDirect(capability) 第一種方式是分配 OS 本地內存, 不屬於 GC 管轄範圍, 由於不需要內存拷貝所以速度相對較塊。
 *
 * 但如果不斷分配本地內存, 堆內存很少使用, 那麼 JVM 就不需要執行 GC, DirectByteBuffer 對象們就不會被回收,
 * 這時候堆內存充足, 但本地內存可能已經使用光了, 再次嘗試分配本地內存就會出現 OutOfMemoryError, 那程序就直接崩潰了。
 */
public class DirectBufferMemoryDemo {
    public static void main(String[] args) {
        System.out.println("配置的 maxDirectMemory:" + (sun.misc.VM.maxDirectMemory() / (double) 1024 / 1024)  + "MB");

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        // -XX:MaxDirectMemorySize=5m   我們配置爲5MB, 但實際使用6MB, 故意使壞
        ByteBuffer bb = ByteBuffer.allocateDirect(6 * 1024 * 1024);
    }
}

🍀 java.lang.OutOfMemoryError: unable to create new native thread

package com.brian.interview.study.jvm.oom;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 14:36
 */

/**
 * 高併發請求服務器時, 經常出現如下異常:java.lang.OutOfMemoryError: unable to create new native thread
 * 準確的講該 native thread 異常與對應的平臺有關
 *
 * 導致原因:
 *      1、你的應用創建了太多線程了, 一個應用進程創建多個線程, 超過系統承載極限
 *      2、你的服務器並不允許你的應用程序創建這麼多線程, linux 系統默認允許單個進程可以創建的線程數是 1024 個,
 *      你的應用創建超過這個數量, 就會報 java.lang.OutOfMemoryError: unable to create new native thread
 *
 * 解決辦法:
 * 1、想辦法降低你應用程序創建線程的數量, 分析應用是否真的需要創建這麼多線程, 如果不是, 改代碼將線程數降到最低
 * 2、對於有的應用, 確定需要創建很多線程, 遠超過 linux 系統的默認 1024 個線程的限制, 可以通過修改 linux 服務器配置, 擴大 linux 默認限制
 */
public class UnableCreateNewThreadDemo {
    public static void main(String[] args) {

        for (int i = 1; ; i++) {
            System.out.println("***************** i=" + i);

            new Thread(() -> {
                try {
                    Thread.sleep(Integer.MAX_VALUE);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }, "" + i).start();
        }

    }
}

📌 非 root 用戶登錄 Linux 系統測試

📌 服務器級別調參調優

vim /etc/security/limits.d/90-nproc.conf
在這裏插入圖片描述
打開後發現除了 root,其他賬號都限制在 1024 個
在這裏插入圖片描述
如果我們想要張三這個用戶運行,希望他生成的線程多一些,我們可以如下配置
在這裏插入圖片描述

🍀 java.lang.OutOfMemoryError: Metaspace

package com.brian.interview.study.jvm.oom;

/**
 * Copyright (c) 2020 ZJU All Rights Reserved
 * <p>
 * Project: JavaSomeDemo
 * Package: com.brian.interview.study.jvm.oom
 * Version: 1.0
 * <p>
 * Created by Brian on 2020/2/15 15:14
 */

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * JVM 參數
 * -XX:MetaspaceSize=8m -XX:MaxMetaspaceSize=8m
 *
 * Java 8 及之後的版本使用 Metaspace 來替代永久代。
 *
 * Metaspace 是方法區在 HotSpot 中的實現, 它與持久代最大的區別在於:Metaspace 並不在虛擬機內存中而是使用本地內存
 * 也即在 java8 中, classes metadata(the virtual machines internal presentation of Java class), 被存儲在叫做
 * Metaspace 的 native memory
 *
 * 永久代(java8 後被元空間 Metaspace 取代了)存放了以下信息:
 *
 *  虛擬機加載的類信息
 *  常量池
 *  靜態變量
 *  即時編譯後的代碼
 *
 * 模擬 Metaspace 空間溢出, 我們不斷生成類往元空間灌, 類佔據的空間總是會超過 Metaspace 指定的空間大小的
 */
public class MetaspaceOOMTest {

    static class  OOMTest{

    }

    public static void main(String[] args) {
        int i = 0;  // 模擬計數多少次以後發生異常

        try {
            while (true) {
                i++;
                Enhancer enhancer = new Enhancer();
                enhancer.setSuperclass(OOMTest.class);
                enhancer.setUseCache(false);
                enhancer.setCallback(new MethodInterceptor() {
                    @Override
                    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                        return methodProxy.invokeSuper(o, args);
                    }
                });
                enhancer.create();
            }
        } catch (Throwable e) {
            System.out.println("***********多少次後發生了異常:" + i);
            e.printStackTrace();
        }
    }
}

// java.lang.OutOfMemoryError: Metaspace

📌 使用 java -XX:+PrintFlagsInitial 命令查看本機的初始化參數,-XX:MetaspaceSize 爲21810376B(大約20.8M)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章