如果线程池中线程异常后:销毁还是复用?

hi,我是阿昌,今天分享一下如果线程池中线程异常后:销毁还是复用?

下面讨论的线程池是:java.util.concurrent.ExecutorService线程池

通过 execute提交任务

测试代码

java 复制代码
public static void main(String[] args) {
  ExecutorService executorService = buildThreadPoolExecutor();
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute-exception"));
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));


  try {
    Thread.sleep(5000);
  } catch (InterruptedException e) {
    throw new RuntimeException(e);
  }
  System.out.println("再次执行任务=======================");

  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));
  executorService.execute(() -> exeTask("achang-execute"));
}

public static ExecutorService buildThreadPoolExecutor() {
  return new ThreadPoolExecutor(5, 10, 30, TimeUnit.SECONDS,
                                new LinkedBlockingQueue<>(1000), new ThreadFactoryBuilder().setNameFormat("achang-test-%s").build()
                                , new ThreadPoolExecutor.CallerRunsPolicy());
}

private static void exeTask(String name) {
  String printStr = "[thread-name:" + Thread.currentThread().getName() + ",执行方式:" + name + "]";
  if ("achang-execute-exception".equals(name)) {
    throw new RuntimeException(printStr + ", 我出现了异常了");
  } else {
    System.out.println(printStr);
  }
}

执行结果

结论

execute 提交到线程池的方式,如果执行中抛出异常,并且没有在执行逻辑中catch,那么会抛出异常,并且移除抛出异常的线程,创建新的线程放入到线程池中。

通过submit提交任务

测试代码

java 复制代码
public static void main(String[] args) {
		ExecutorService executorService = buildThreadPoolExecutor();
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit-exception"));
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
		
		
		try {
			Thread.sleep(5000);
		} catch (InterruptedException e) {
			throw new RuntimeException(e);
		}
		System.out.println("再次执行任务=======================");
		
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
		executorService.submit(() -> exeTask("achang-submit"));
	}
	
	public static ExecutorService buildThreadPoolExecutor() {
		return new ThreadPoolExecutor(5, 10, 30, TimeUnit.SECONDS,
				new LinkedBlockingQueue<>(1000), new ThreadFactoryBuilder().setNameFormat("achang-test-%s").build()
				, new ThreadPoolExecutor.CallerRunsPolicy());
	}
	
	private static void exeTask(String name) {
		String printStr = "[thread-name:" + Thread.currentThread().getName() + ",执行方式:" + name + "]";
		if ("achang-execute-exception".equals(name)) {
			throw new RuntimeException(printStr + ", 我出现了异常了");
		} else {
			System.out.println(printStr);
		}
	}

执行结果

结论

submit 提交到线程池的方式,如果执行中抛出异常,并且没有catch,不会抛出异常,不会创建新的线程。

源码解析

java.util.concurrent.AbstractExecutorService#submit(java.lang.Runnable)

上面的execute会调用到java.util.concurrent.ThreadPoolExecutor#runWorker:

java.util.concurrent.ThreadPoolExecutor#processWorkerExit

上面可以看出,如果抛出异常,会移除抛出异常的线程,创建新的线程。

那为啥submit()方法,没有创建新的线程,而是继续复用原线程?

看源码可以知道submit也是调用了execute方法,但是在调用之前,包装了一层 RunnableFuture,那一定是在RunnableFuture的实现 FutureTask中有特殊处理了,如下查看源码:

java.util.concurrent.FutureTask#run

java.util.concurrent.FutureTask#setException

java.util.concurrent.FutureTask#get()

获取执行结果会调用report()

java.util.concurrent.FutureTask#report

发现通过s状态位来判断任务状态,来判断是否抛出异常

上面看到通过java.util.concurrent.FutureTask#get(),就可以获取对应的异常信息。

总结

若当线程池里的线程出现了异常后:

  • 如果调用方式是execute时,可以看到堆栈异常的输出,线程池会把这个线程移除掉,并创建一个新的线程放到线程池中。
  • 如果调用是submit时,堆栈异常没有输出。但是调用Future.get()方法时,可以捕获到异常,不会把这个线程移除掉,也不会创建新的线程放入到线程池中

上面2种调用方式,都不会影响到线程池里面其他线程的正常执行。

相关推荐
2501_947575809 小时前
计算机毕业设计之jsp开山车行二手车交易系统
java·开发语言·hadoop·python·信息可视化·django·课程设计
骑士雄师9 小时前
java面试题 4:鉴权
java·开发语言
帅次11 小时前
Android 高级工程师面试:Java 基础知识 近1年高频追问 22 题
android·java·面试
蓝胖的四次元口袋11 小时前
Java集合(4)
java·哈希算法
2501_9481069111 小时前
计算机毕业设计之基于jsp教科研信息共享系统
java·开发语言·信息可视化·spark·课程设计
TanYYF11 小时前
spring ai入门教程二
java·人工智能·spring
SeeYa-J11 小时前
Spring IOC(Inversion of Control)
java·spring·rpc
不会c+12 小时前
02-SpringBoot配置文件
java·spring boot·后端
AI 大模型学习不踩坑12 小时前
OpenClaw 完整教程:从安装到使用(官方脚本版)
java·人工智能·神经网络·机器学习·计算机视觉·自然语言处理·openclaw
Listen·Rain13 小时前
数据库流式查询
java·数据库