OutOfMemoryError/OOM/内存溢出异常实例分析--虚拟机栈和本地方法栈溢出

时间:2022-12-27 17:35:15

关于虚拟机栈和本地方法栈,在JVM规范中描述了两种异常:

1.如果线程请求的栈深度大于JVM所允许的深度,将抛出*Error异常;

2.如果虚拟机在扩展栈时无法申请到足够的内存,就会抛出OutOfMemoryError异常。

下面进行虚拟机栈和本地方法栈的SOF异常测试:

public class JavaVMStackSOF {

    private int stackLenth = 1;

    public void stackLeak() {
        stackLenth++;
        stackLeak();
    }

    public static void main(String[] args) throws Throwable{
        JavaVMStackSOF oom = new JavaVMStackSOF();
        try {
            oom.stackLeak();
        } catch (Throwable e) {
            System.out.println("栈深度:" + oom.stackLenth);
            throw e;
        }
    }
}

运行时设置栈容量为:-Xss128k

具体操作可参照:OutOfMemoryError/OOM/内存溢出异常实例分析--堆内存溢出

运行结果如下:

OutOfMemoryError/OOM/内存溢出异常实例分析--虚拟机栈和本地方法栈溢出

实验结果表明:在单线程下,无论是 由于栈帧太大还是虚拟机栈容量太小,当内存无法分配的时候,虚拟机栈跑出的都是*Error异常。

怎么产生OOM异常呢?

可以通过不断创建线程的方法,在这种情况下,为每个线程分配的内存越大,就会越容易产生OOM异常。

下面在测试之前,先特别提示一下,如果想测试栈的OOM异常,记得先保存当前的工作。由于Windows平台的虚拟机中,Java的线程是映射到操作

系统的内核上的,因此以下代码可能会导致操作系统假死

public class JavaVMStackOOM {

    private void dontStop() {
        while (true) {
        }
    }

    public void stackLeakByThread() {
        while (true) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    dontStop();
                }
            }).start();
        }
    }

    public static void main(String[] args) {
        JavaVMStackOOM oom = new JavaVMStackOOM();
        oom.stackLeakByThread();
    }
}

参数设置如下:

OutOfMemoryError/OOM/内存溢出异常实例分析--虚拟机栈和本地方法栈溢出

我测了一下,系统差点卡死,就不展示测试结果了