在 Java 中,线程池的状态和线程的状态是齐全不同的,线程有 6 种状态:NEW:初始化状态、RUNNABLE:可运行/运行状态、BLOCKED:阻塞状态、WAITING:无时限期待状态、TIMED_WAITING:有时限期待状态和 TERMINATED:终止状态。而线程池的状态有以下 5 种:

  1. RUNNING:运行状态,线程池创立好之后就会进入此状态,如果不手动调用敞开办法,那么线程池在整个程序运行期间都是此状态。
  2. SHUTDOWN:敞开状态,不再承受新工作提交,然而会将已保留在工作队列中的工作解决完。
  3. STOP:进行状态,不再承受新工作提交,并且会中断以后正在执行的工作、放弃工作队列中已有的工作。
  4. TIDYING:整顿状态,所有的工作都执行结束后(也包含工作队列中的工作执行完),以后线程池中的流动线程数降为 0 时的状态。到此状态之后,会调用线程池的 terminated() 办法。
  5. TERMINATED:销毁状态,当执行完线程池的 terminated() 办法之后就会变为此状态。

这 5 种状态能够在 ThreadPoolExecutor 源码中找到,如下图所示:

线程池状态转移

线程池的状态转移有两条门路:

  1. 当调用 shutdown() 办法时,线程池的状态会从 RUNNING 到 SHUTDOWN,再到 TIDYING,最初到 TERMENATED 销毁状态。
  2. 当调用 shutdownNow() 办法时,线程池的状态会从 RUNNING 到 STOP,再到 TIDYING,最初到 TERMENATED 销毁状态。

线程状态转换的流程如下图所示:

terminated办法

线程池中的 terminated() 办法,也就是线程池从 TIDYING 转换到 TERMINATED 状态时调用的办法,默认是空的,它的源码如下:

咱们能够在创立线程池的时候重写 terminated() 办法,具体实现代码如下:

import java.util.concurrent.LinkedBlockingQueue;import java.util.concurrent.ThreadPoolExecutor;import java.util.concurrent.TimeUnit;public class ThreadPoolStateTransition {    public static void main(String[] args) throws InterruptedException {        // 创立线程池        ThreadPoolExecutor threadPool = new ThreadPoolExecutor(10, 10, 0L,                TimeUnit.SECONDS, new LinkedBlockingQueue<>(100)) {            @Override            protected void terminated() {                super.terminated();                System.out.println("执行 terminated() 办法");            }        };        // 敞开线程池        threadPool.shutdown();        // 期待线程池执行完再退出        while (!threadPool.awaitTermination(1, TimeUnit.SECONDS)) {            System.out.println("线程池正在运行中");        }    }}

总结

线程池的状态总共有 5 种:RUNNING:运行状态、SHUTDOWN:敞开状态、STOP:进行状态、TIDYING:整顿状态和 TERMINATED:销毁状态。默认状况下,如果不调用敞开办法,线程池会始终处于 RUNNING 状态,而线程池状态的转移有两个门路:当调用 shutdown() 办法时,线程池的状态会从 RUNNING 到 SHUTDOWN,再到 TIDYING,最初到 TERMENATED 销毁状态;当调用 shutdownNow() 办法时,线程池的状态会从 RUNNING 到 STOP,再到 TIDYING,最初到 TERMENATED 销毁状态。

是非审之于己,毁誉听之于人,得失安之于数。

公众号:Java面试真题解析

面试合集:https://gitee.com/mydb/interview