一个线程OOM之后,进程会崩溃么

偶然间看到一个问题,一个线程OOM之后,进程里面的其他线程还能运行吗?看了其他博客里面都说可以运行,但是按照项目上的经验,我们的项目是部署在tomcat中运行,如果发生OOM,这个时候发送请求是不会有回应的!所以本文打算分析一下某个线程内存溢出是,进程中的其他线程到低能否正常运行。

 

先说下个人测试得出来的结论:

OutOfMemoryError是一种错误,它是 JVM 的一种自我防御机制,用于防止整个应用程序崩溃,这个时候线程会被Kill掉,然后抛出OutOfMemory异常信息。因为栈上的空间是线程私有的,所以线程被Kill掉之后,栈上的空间就被释放了,但是堆空间是共享的,被Kill掉的线程中的对象可能被该线程之外的其他线程引用,这个时候这部分对象就没有办法被GC掉,其他线程如果此时需要申请资源但是又资源又不足,那么此时其他线程就不能运行

为什么tomcat内存溢出之后其他请求也无法响应了呢?个人猜测是:OutOfMemory线程所持有的大对象被其他线程引用了,所以GC的时候GC Root链还是可达的,同时其他线程也要使用内存,此时内存不够了,所以其他线程也用不了了。

要明确的一点是:线程被Kill和内存回收是两件完全不同的事情

 

个人测试代码如下:

    public static void main(String[] args) {

//        //todo list加在这儿
//        List<byte[]> list = new ArrayList<>();


        Thread thread1 = new Thread(new Runnable() {
            //todo list加在这儿
            List<byte[]> list = new ArrayList<>();

            @Override
            public void run() {
                while (true) {
                    System.out.println("Thread1 : now list size is : " + list.size());
                    byte[] b = new byte[1024 * 1024];
                    list.add(b);

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

                }
            }
        });


        Thread thread2 = new Thread(() -> {
            while (true) {
                System.out.println("Hello, this Thread steal alived ");
                try {
                    TimeUnit.MILLISECONDS.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });


        //这个线程也在申请资源
        Thread thread3 = new Thread(new Runnable() {
            //list加在这儿
            List<byte[]> list = new ArrayList<>();

            @Override
            public void run() {
                while (true) {
                    System.out.println("Thread3 : now list size is : " + list.size());
                    byte[] b = new byte[1024 * 1024];
                    list.add(b);

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


        thread1.start();
        thread2.start();
        thread3.start();

        while (true) {
            try {
                TimeUnit.SECONDS.sleep(1);
                System.out.println("Main Thread start GC");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.gc();
        }


    }

Thread1和Thread3都在申请内存,Thread2只是输出信息。

 

  1. 开启Thread1和Thread2,不显示调用System.gc(),变量为线程是有或者被主线程持有

线程会被Kill,但是不管变量是否是线程私有的,内存都不会被回收,因为不满足GC条件。图示如下:

 

2.开始Thread1和Thread2,显示调用System.gc(),变量如果为线程私有的话会被回收,如果还被其他线程持有,那么这部分空间就不会被回收

引用还被其他线程持有的情况:

对象为线程私有的情况:

    可以看出,如果对象的引用被其他线程持有,对象是不会被回收的!

 

3.三个线程都开启,不显示调用System.gc(),对象未线程私有

       Thread1和Thread3会先后发生线程溢出,Thread1在先溢出之后,对象没有被回收,但是Thread3也在持续申请内存。所以Thread3申请不到内存时,JVM会进行一次Full GC,回收Thread1占用的内存好分配给Thread3。Thread3一直申请内存直到内存溢出。

 

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值