线程池ShutDown和ShowDownNow区别

时间:2022-07-24
本文章向大家介绍线程池ShutDown和ShowDownNow区别,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

在写上一篇文章的时候就在想,executor.shutDown会不会把正在执行中的任务给中断了?虽然当时也分析过,但是当时确实只是泛泛而读,在实际中还是相当的模糊。借此机会,我们再复习一下。

    public void shutdown() {
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            checkShutdownAccess();
            //将线程池状态设置为shutdown
            advanceRunState(SHUTDOWN);
            //中断线程池中的worker线程
            interruptIdleWorkers();
            onShutdown(); // hook for ScheduledThreadPoolExecutor
        } finally {
            mainLock.unlock();
        }
        tryTerminate();
    }

那么中断线程的操作就在这个interruptIdleWorkers()中。

    private void interruptIdleWorkers(boolean onlyOne) {
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            for (Worker w : workers) {
                Thread t = w.thread;
                //这里的tryLock是尝试获取工作线程的锁,如果工作线程在运行
                //那么就获取不到锁,也就是无法中断。所以这里中断的都是没有运行的线程
                if (!t.isInterrupted() && w.tryLock()) {
                    try {
                        //调用每个线程的中断
                        t.interrupt();
                    } catch (SecurityException ignore) {
                    } finally {
                        w.unlock();
                    }
                }
                if (onlyOne)
                    break;
            }
        } finally {
            mainLock.unlock();
        }
    }

所以说shutdown是不会让线程直接退出的,之前添加的线程还是会运行到自然结束。但是因为这块设置了线程池状态为shutdown,因此线程是添加不了的。

而在shutdownNow方法中

    public List<Runnable> shutdownNow() {
        List<Runnable> tasks;
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            checkShutdownAccess();
            //直接设置为stop状态
            advanceRunState(STOP);
            //中断线程
            interruptWorkers();
            tasks = drainQueue();
        } finally {
            mainLock.unlock();
        }
        tryTerminate();
        return tasks;
    }
    private void interruptWorkers() {
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            for (Worker w : workers)
               //逐个进行中断
                w.interruptIfStarted();
        } finally {
            mainLock.unlock();
        }
    }
        void interruptIfStarted() {
            Thread t;
            if (getState() >= 0 && (t = thread) != null && !t.isInterrupted()) {
                try {
                //直接调用中断
                    t.interrupt();
                } catch (SecurityException ignore) {
                }
            }
        }

所以看的出来,ShutDownNow是直接中断所有的工作线程。并且阻止新线程的添加。