当前位置: 首页 > article >正文

“线程池中线程异常后:销毁还是复用?”

目录

一、验证execute提交线程池中

测试

结论

二、验证submit提交线程池中

测试

结论

三、源码解析

查看submit方法的执行逻辑

查看execute方法的执行逻辑

为什么submit方法,没有创建新的线程,而是继续复用原线程?

四、总结


需要说明,本文的线程池都是java.util.concurrent.ExecutorService线程池,本文将围绕验证,阅读源码俩方面来解析这个问题。

一、验证execute提交线程池中

测试

我们首先定义了一个 ThreadPoolExecutorDeadTest 类,用来演示 Java 中 ThreadPoolExecutor 的使用,以及在任务执行过程中发生异常时线程池的行为。然后,在代码中,通过创建一个线程池并提交多个任务来模拟线程池的运行情况,其中一个任务故意抛出异常。

测试代码如下:

public class ThreadPoolExecutorDeadTest {
  public static void main(String[] args) throws InterruptedException {
    ExecutorService executorService = buildThreadPoolExecutor();
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute-exception"));
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
    Thread.sleep(5000);
    System.out.println("再次执行任务=======================");
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
    executorService.execute(() -> exeTask("execute"));
  }

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

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

首先,调用 buildThreadPoolExecutor() 方法创建一个线程池并赋值给 executorService。这个线程池具有核心线程数 5,最大线程数 10,空闲线程存活时间 30 秒,任务队列容量为 1000,使用 CallerRunsPolicy 作为拒绝策略。 

然后,调用 executorService.execute() 方法多次向线程池提交任务。任务类型为 Runnable,任务的内容是调用 exeTask 方法。

主线程在前五个任务提交后,调用 Thread.sleep(5000) 暂停5秒,以模拟任务执行间隔。在暂停后,向线程池再次提交五个任务,继续执行。

运行结果如下, 

可以看到test-2线程不见了,出现了test-3线程。

结论

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

二、验证submit提交线程池中

测试

测试代码:

public class ThreadPoolExecutorDeadTest {
  public static void main(String[] args) throws InterruptedException {
    ExecutorService executorService = buildThreadPoolExecutor();
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute-exception"));
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
    Thread.sleep(5000);
    System.out.println("再次执行任务=======================");
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
    executorService.submit(() -> exeTask("execute"));
  }
  
  public static ExecutorService buildThreadPoolExecutor() {
    return new ThreadPoolExecutor(5, 10, 30, TimeUnit.SECONDS,
      new LinkedBlockingQueue<>(1000), new ThreadFactoryBuilder().setNameFormat("test-%s").build()
      , new ThreadPoolExecutor.CallerRunsPolicy());
  }

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

运行结果如下,

可以看到test-2线程执行异常了,并没有抛出异常栈而且继续复用了,没有被移除。 

结论

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

 

三、源码解析

查看submit方法的执行逻辑

我们从上述测试代码中的executorService.submit(() -> exeTask("execute"));,进入到 java.util.concurrent.AbstractExecutorService#submit(java.lang.Runnable)方法,可以看到,submit方法,也是包装了RunnableFuture后调用execute方法。

查看execute方法的执行逻辑

进入execute方法,然后进入execute方法的实现方法,就可以进入到java.util.concurrent.ThreadPoolExecutor#runWorker方法,

可以发现,run方法抛出异常之后,会执行finally。

然后,我们进入到 java.util.concurrent.ThreadPoolExecutor#processWorkerExit方法,

可以发现,如果抛出异常,会移除抛出异常的线程,创建新的线程。

为什么submit方法,没有创建新的线程,而是继续复用原线程?

还记得,我们在查看submit方法的执行逻辑的时候,发现submit也是调用了execute方法,但是在调用之前,包装了一层 RunnableFuture,那一定是在RunnableFuture的实现 FutureTask中有特殊处理了,我们查看源码可以发现。

进入 FutureTask的run方法,

可以发现包装的一层task,catch了异常,并没有往上抛。所以不会移除抛出异常的线程,创建新的线程。

但是,我们通过java.util.concurrent.FutureTask#get(),就可以获取对应的异常信息。

 

四、总结

当一个线程池里面的线程异常后:

  • 当执行方式是execute()时,可以看到堆栈异常的输出,线程池会把这个线程移除掉,并创建一个新的线程放到线程池中。

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

以上俩种执行方式,都不会影响线程池里面其他线程的正常执行。


http://www.kler.cn/a/286246.html

相关文章:

  • 【PLL】杂散生成和调制
  • Springboot使用AOP时,需不需要引入AspectJ?
  • 【C++】类与对象(下)
  • 商品列表及商品详情展示
  • 【算法】动态规划专题① ——线性DP python
  • deepseek R1 14b显存占用
  • OpenAI 神秘模型「草莓」预计今秋推出,ChatGPT 将迎重大升级|TodayAI
  • 【60天备战软考高级系统架构设计师——第五天:需求分析方法与工具】
  • uniapp组件用法
  • Android自定义View实现不同朝向字体变色
  • Leetcode 3272. Find the Count of Good Integers
  • RabbitMQ 是什么?应用场景有哪些?
  • IBM Speech to Text:发出语音识别请求
  • Qt 实现应用程序换肤功能
  • ASP.NET Core 入门教程二 实现基本 GET 和 POST 接口
  • 【论文解读】SAM模型超级进化:面向移动端的轻量级SAM,比FastSAM快4倍!(附论文地址)
  • 【攻略】第三届数据库大赛创新上云性能挑战赛-高性能分析型查询引擎赛道-冠军
  • OpenCV绘图函数(5)绘制标记函数drawMarker()的使用
  • C++避坑小知识
  • 短视频流量|基于SprinBoot+vue的短视频流量数据分析系统(源码+数据库+文档)
  • 【华三】不懂链路聚合?看这篇就够了!华三配置详解
  • 公众号里的产品宣传册是如何制作的?
  • 2024HarmonyOS应用开发者高级认证最新整理题库和答案(已收录182道 )
  • 【Qt的TS文件转换器】利用Python实现自动化TS文件转换
  • 疲劳驾驶行为检测检测系统源码分享 # [一条龙教学YOLOV8标注好的数据集一键训练_70+全套改进创新点发刊_Web前端展示]
  • DNS部署与安全