我也来说说经典面试题目-“OOM异常会导致JVM退出吗?”

经典面试题目"OOM异常会导致JVM退出吗?

我的回答是"这要分情况看,对于守护线程来说,OOM并不会导致JVM退出;对于非守护线程来说,如果某个线程捕获了OOM异常并处理异常后线程并未退出,那JVM并不会退出;如果线程没有捕获Error异常,那么将由全局的异常处理器处理,默认的全局的异常处理器也会让当前这个发生异常的线程退出,但是如果这个线程是最后一个非守护线程,那么JVM会退出,如果不是,JVM并不会退出。

对于守护线程来说,OOM并不会导致JVM退出,这里有一个非常好的线上故障:https://blog.csdn.net/shuxiaohua/article/details/114658325 ,缺少的接收客户端请求的线程Acceptor是一个守护线程,并且因为OOM退出时,并没有让Tomcat退出。

下面介绍一下非守护线程的情况,这些非守护线程通常就是处理业务的线程。每一个写过Java的人都应该知道Java异常继承体系,如下图所示。

这里我们要注意一点,就是Exception和Error有共同的父类Throwable,这意味着异常和错误都可以在Java层捕获,例如:

复制代码
public static void main(String[] args) throws InterruptedException {
    try{
        // 每个整数数组的大小为4M
        int[] array1 = new int[1_000_000];
        int[] array2 = new int[1_000_000];
        int[] array3 = new int[1_000_000];
        int[] array4 = new int[1_000_000];
        int[] array5 = new int[1_000_000];
    }catch (Throwable t){
        t.printStackTrace();
    }
    System.out.println("程序走到了这里!");
}

我们在指定参数-Xms20M -Xmx20m后,运行打印结果如下:

复制代码
java.lang.OutOfMemoryError: Java heap space
	at cn.hotspotvm.TestError.main(TestError.java:12)
程序走到了这里!

可以看到,即使发生了错误,这个线程依然在正常运行,如果我们不对Error进行捕获呢?如下:

复制代码
public static void main(String[] args) throws InterruptedException {
    int[] array1 = new int[1_000_000];
    int[] array2 = new int[1_000_000];
    int[] array3 = new int[1_000_000];
    int[] array4 = new int[1_000_000];
    int[] array5 = new int[1_000_000];
    System.out.println("程序走到了这里!");
}

再次运行后就的打印结果如下:

复制代码
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
	at cn.hotspotvm.TestError.main(TestError.java:10)

可以看到,在发生错误时这个线程没有走到打印语句,而是直接退出了。

这里其实要说明的是,我们可以设置一个全局的异常处理器来统一处理,或者优先针对某个线程设置异常处理器,这样当我们忽略了捕获错误时,可以在全局异常处理器中进行处理。举个例子如下:

复制代码
public static void main(String[] args) throws InterruptedException {
    Thread.setDefaultUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
        @Override
        public void uncaughtException(Thread t, Throwable e) {
            System.out.println("这里是全局异常处理 "+e.getLocalizedMessage());
        }
    });

    int[] array1 = new int[1_000_000];
    int[] array2 = new int[1_000_000];
    int[] array3 = new int[1_000_000];
    int[] array4 = new int[1_000_000];
    int[] array5 = new int[1_000_000];

    System.out.println("程序走到了这里!");
}

打印的信息如下:

复制代码
这里是全局异常处理 Java heap space

这个线程在全局异常处理中如果没有特殊处理,通常会让当前的线程退出。如果当前线程退出,那么JVM会退出吗?

其实Java虚拟机退出的条件是:虚拟机内已经没有了非守护线程。线程发生未处理的异常最终导致线程结束时,如果这个线程是最后一个非守护线程,则会退出,否则不退出。

更多文章可访问:JDK源码剖析网