站点图标 度崩网-几度崩溃

如何停止一个线程池?

Java 并发工具包中java.util.concurrent.ExecutorService 接口定义了线程池任务提交、获取线程池状态、线程池停止的方法等。

JDK 1.8 中,线程池的停止一般使用shutdown()、shutdownNow()、shutdown() + awaitTermination(long timeout, TimeUnit unit) 方法。

1、shutdown() 方法源码中解释

 * Initiates an orderly shutdown in which previously submitted * tasks are executed, but no new tasks will be accepted. * Invocation has no additional effect if already shut down.

2、shutdownNow() 方法源码中解释

 * Attempts to stop all actively executing tasks, halts the * processing of waiting tasks, and returns a list of the tasks * that were awaiting execution.

3、awaitTermination(long timeout, TimeUnit unit) 方法源码中解释

 * Blocks until all tasks have completed execution after a shutdown * request, or the timeout occurs, or the current thread is * interrupted, whichever happens first. * * @param timeout the maximum time to wait * @param unit the time unit of the timeout argument * @return {@code true} if this executor terminated and * {@code false} if the timeout elapsed before termination * @throws InterruptedException if interrupted while waiting

实践:

1、使用Executors.newFixedThreadPool(int nThreads) 创建固定大小线程池,测试shutdown() 方法

package constxiong.concurrency.a013;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;/** * 测试固定数量线程池 shutdown() 方法 * @author ConstXiong */public class TestFixedThreadPoolShutdown {public static void main(String[] args) {//创建固定 3 个线程的线程池ExecutorService threadPool = Executors.newFixedThreadPool(3);//向线程池提交 10 个任务for (int i = 1; i <= 10; i++) {final int index = i;threadPool.submit(() -> {System.out.println("正在执行任务 " + index);//休眠 3 秒try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}});}//休眠 4 秒try {Thread.sleep(4000);} catch (InterruptedException e) {e.printStackTrace();}//关闭线程池threadPool.shutdown();}}

打印结果如下,可以看出,主线程向线程池提交了 10 个任务,休眠 4秒后关闭线程池,线程池把 10 个任务都执行完成后关闭了。

正在执行任务 1正在执行任务 3正在执行任务 2正在执行任务 4正在执行任务 6正在执行任务 5正在执行任务 8正在执行任务 9正在执行任务 7正在执行任务 10

2、使用Executors.newFixedThreadPool(int nThreads) 创建固定大小线程池,测试shutdownNow() 方法

package constxiong.concurrency.a013;import java.util.List;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;/** * 测试固定数量线程池 shutdownNow() 方法 * @author ConstXiong */public class TestFixedThreadPoolShutdownNow {public static void main(String[] args) {//创建固定 3 个线程的线程池ExecutorService threadPool = Executors.newFixedThreadPool(3);//向线程池提交 10 个任务for (int i = 1; i <= 10; i++) {final int index = i;threadPool.submit(() -> {System.out.println("正在执行任务 " + index);//休眠 3 秒try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}});}//休眠 4 秒try {Thread.sleep(4000);} catch (InterruptedException e) {e.printStackTrace();}//关闭线程池List<Runnable> tasks = threadPool.shutdownNow();System.out.println("剩余 " + tasks.size() + " 个任务未执行");}}

打印结果如下,可以看出,主线程向线程池提交了 10 个任务,休眠 4秒后关闭线程池,线程池执行了 6 个任务,抛出异常,打印返回的剩余未执行的任务个数。

正在执行任务 1正在执行任务 2正在执行任务 3正在执行任务 4正在执行任务 6正在执行任务 5剩余 4 个任务未执行java.lang.InterruptedException: sleep interruptedat java.lang.Thread.sleep(Native Method)at constxiong.concurrency.a013.TestFixedThreadPoolShutdownNow.lambda$0(TestFixedThreadPoolShutdownNow.java:24)at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)at java.util.concurrent.FutureTask.run(FutureTask.java:266)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)at java.lang.Thread.run(Thread.java:748)java.lang.InterruptedException: sleep interruptedat java.lang.Thread.sleep(Native Method)at constxiong.concurrency.a013.TestFixedThreadPoolShutdownNow.lambda$0(TestFixedThreadPoolShutdownNow.java:24)at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)at java.util.concurrent.FutureTask.run(FutureTask.java:266)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)at java.lang.Thread.run(Thread.java:748)java.lang.InterruptedException: sleep interruptedat java.lang.Thread.sleep(Native Method)at constxiong.concurrency.a013.TestFixedThreadPoolShutdownNow.lambda$0(TestFixedThreadPoolShutdownNow.java:24)at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)at java.util.concurrent.FutureTask.run(FutureTask.java:266)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)at java.lang.Thread.run(Thread.java:748)

3、Executors.newFixedThreadPool(int nThreads) 创建固定大小线程池,测试awaitTermination(long timeout, TimeUnit unit) 方法

package constxiong.concurrency.a013;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.TimeUnit;/** * 测试固定数量线程池 shutdownNow() 方法 * @author ConstXiong */public class TestFixedThreadPoolAwaitTermination {public static void main(String[] args) {//创建固定 3 个线程的线程池ExecutorService threadPool = Executors.newFixedThreadPool(3);//向线程池提交 10 个任务for (int i = 1; i <= 10; i++) {final int index = i;threadPool.submit(() -> {System.out.println("正在执行任务 " + index);//休眠 3 秒try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}});}//关闭线程池,设置等待超时时间 3 秒System.out.println("设置线程池关闭,等待 3 秒...");threadPool.shutdown();try {boolean isTermination = threadPool.awaitTermination(3, TimeUnit.SECONDS);System.out.println(isTermination ? "线程池已停止" : "线程池未停止");} catch (InterruptedException e) {e.printStackTrace();}//再等待超时时间 20 秒System.out.println("再等待 20 秒...");try {boolean isTermination = threadPool.awaitTermination(20, TimeUnit.SECONDS);System.out.println(isTermination ? "线程池已停止" : "线程池未停止");} catch (InterruptedException e) {e.printStackTrace();}}}

打印结果如下,可以看出,主线程向线程池提交了 10 个任务,申请关闭线程池 3秒超时,3 秒后线程池并未成功关闭;再获取线程池关闭状态20秒超时,线程池成功关闭。

正在执行任务 1正在执行任务 3正在执行任务 2设置线程池关闭,等待 3 秒...线程池未停止正在执行任务 4正在执行任务 6再等待 20 秒...正在执行任务 5正在执行任务 7正在执行任务 9正在执行任务 8正在执行任务 10线程池已停止

总结:

  1. 调用 shutdown() 和shutdownNow() 方法关闭线程池,线程池都无法接收新的任务
  2. shutdown() 方法会继续执行正在执行未完成的任务;shutdownNow() 方法会尝试停止所有正在执行的任务
  3. shutdown() 方法没有返回值;shutdownNow() 方法返回等待执行的任务列表
  4. awaitTermination(long timeout, TimeUnit unit) 方法可以获取线程池是否已经关闭,需要配合shutdown() 使用
  5. shutdownNow() 不一定能够立马结束线程池,该方法会尝试停止所有正在执行的任务,通过调用 Thread.interrupt() 方法来实现的,如果线程中没有 sleep() 、wait()、Condition、定时锁等应用, interrupt() 方法是无法中断当前的线程的。