singleThreadExecutor在线程异常结束时,会有另一个线程取代的实现方式

来源:9-7 Java线程池

斗毂於菟

2021-09-13

老师说的newSingleThreadExecutor()单线程池在线程异常结束时,会有另一个线程取代,而且源码注释中确实有(if this single thread terminates due to a failure during execution prior to shutdown, a new one will take its place if needed to execute subsequent tasks),但是newFixedThreadPool()也有相同意思的注释(If any thread terminates due to a failure during execution prior to shutdown, a new one will take its place if needed to execute subsequent tasks.),看源码中newSingleThreadExecutor()和newFixedThreadPool()的区别,仅仅是newSingleThreadExecutor()返回的线程池少暴露了一些方法并且多了个finalize方法保证线程池调用shutdown(),所以没看明白这个是怎么实现的。后面用代码测试了一下

public static void main(String[] args) {
        ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();
        //Thread.setDefaultUncaughtExceptionHandler(new ThreadExceptionHandler("线程异常捕获器", singleThreadExecutor));

        singleThreadExecutor.execute(new Runnable() {
            private Integer index = 0;
            @Override
            public void run() {
                while (true) {
                    index++;
                    try {
                        TimeUnit.SECONDS.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    if (index == 2) {
                        throw new IllegalStateException("handler exception");
                    }
                    System.out.println(Thread.currentThread().getName() + "NO." + index);
                }
            }
        });
    }

第一次测试我只是在运行中抛出一个异常,由于run()方法不能再抛出异常,线程异常中止,但是并没有另一个线程去取代异常终止的线程去完成剩下的任务。

// 异常捕获
@Override
    public void uncaughtException(Thread thread, Throwable throwable) {
        System.out.println(thread.getName() + " thread aborted abnormally .");
        System.out.println(name + "catch exception :" +  thread.getName() + "exception:" + throwable);
        singleThreadExecutor.execute(new testThread(0));
    }

之后我加上了异常捕获器,并且在捕获异常后,再一次用之前的线程池去启动线程,达到目的了

执行结果
pool-1-thread-1NO.1
pool-1-thread-1 thread aborted abnormally .
线程异常捕获器catch exception :pool-1-thread-1exception:java.lang.IllegalStateException: handler exception
pool-1-thread-2NO.1
pool-1-thread-2 thread aborted abnormally .
线程异常捕获器catch exception :pool-1-thread-2exception:java.lang.IllegalStateException: handler exception
pool-1-thread-3NO.1
pool-1-thread-3 thread aborted abnormally .
线程异常捕获器catch exception :pool-1-thread-3exception:java.lang.IllegalStateException: handler exception
pool-1-thread-4NO.1
pool-1-thread-4 thread aborted abnormally .
线程异常捕获器catch exception :pool-1-thread-4exception:java.lang.IllegalStateException: handler exception

所以想问一下老师,singleThreadExecutor是不是需要结合异常捕获的机制,去实现 ” 在线程异常结束时,有另一个线程取代 “ 。

写回答

1回答

翔仔

2021-09-14

同学好,这个注释确实有点歧义,我这边也头一次去了解,它的释义翻译过来是

"

如果任何线程正常关闭之前在执行过程中因失败而提前终止,那么如果有未被执行的后续任务,则会创建新的线程来继续执行

"

我理解因为加入了对当前线程的异常捕获,才能让线程池感知到执行过程中因失败而提前终止了,所以才创建了新的线程去处理

0
1
斗毂於菟
非常感谢!
2021-09-15
共1条回复

剑指Java面试-Offer直通车 百度资深面试官授课

招聘季即将到来,让百度资深面试官来为你的高薪Offer保驾护航

8441 学习 · 1872 问题

查看课程