ExecutorService等待线程完成后优雅结束

2023-05-16

1.概述

该ExecutorService框架可以很容易地在处理多线程任务。我们将举例说明我们等待线程完成执行的一些场景。

此外,我们将展示如何正常关闭ExecutorService并等待已经运行的线程完成其执行。

2.Executor关闭后

使用Executor时,我们可以通过调用shutdown()或shutdownNow()方法将其关闭。虽然,它不会等到所有线程都停止执行。

等待现有线程完成执行可以通过使用awaitTermination()方法来实现。

这会阻塞线程,直到所有任务完成执行或达到指定的超时:

public void awaitAfterShutdown(ExecutorService threadPool) {

    threadPool.shutdown();
    try {
        if (!threadPool.awaitTermination(60, TimeUnit.SECONDS)) {
            threadPool.shutdownNow();
        }
    } catch (InterruptedException ex) {
        threadPool.shutdownNow();
        Thread.currentThread().interrupt();
    }

}

3.使用CountDownLatch

接下来,让我们看看解决此问题的另一种方法 - 使用CountDownLatch来表示任务的完成。

我们可以使用一个值来初始化它,该值表示在通知所有调用await()方法的线程之前可以递减的次数。

例如,如果我们需要当前线程等待另外N个线程完成执行,我们可以使用N初始化锁存器:

ExecutorService WORKER_THREAD_POOL = Executors.newFixedThreadPool(10);
CountDownLatch latch = new CountDownLatch(2);
for(int i = 0;i< 2; i++){
    WORKER_THREAD_POOL.submit(() -> {
        try {
            // ...
            latch.countDown();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    });
}
// wait for the latch to be decremented by the two remaining threads
latch.await();

4.使用invokeAll()

我们可以用来运行线程的第一种方法是invokeAll()方法。在所有任务完成或超时到期后,该方法返回Future对象列表。

此外,我们必须注意返回的Future对象的顺序与提供的Callable对象的列表相同:

ExecutorService WORKER_THREAD_POOL = Executors.newFixedThreadPool(10);

List<Callable<String>> callables = Arrays.asList(new DelayedCallable("fast thread", 100),new DelayedCallable("slow thread", 3000));

long startProcessingTime = System.currentTimeMillis();
List<Future<String>> futures = WORKER_THREAD_POOL.invokeAll(callables);
awaitTerminationAfterShutdown(WORKER_THREAD_POOL);

long totalProcessingTime = System.currentTimeMillis() - startProcessingTime;
assertTrue(totalProcessingTime >= 3000);

String firstThreadResponse = futures.get(0).get();
assertTrue("fast thread".equals(firstThreadResponse));

String secondThreadResponse = futures.get(1).get();
assertTrue("slow thread".equals(secondThreadResponse));

5.使用ExecutorCompletionService

运行多个线程的另一种方法是使用ExecutorCompletionService。它使用提供的ExecutorService来执行任务。

与invokeAll()的一个区别是返回表示执行任务的Futures的顺序。ExecutorCompletionService使用队列按结束顺序存储结果,而invokeAll()返回一个列表,该列表具有与给定任务列表的迭代器生成的顺序相同的顺序:


CompletionService<String> service = new ExecutorCompletionService<>(WORKER_THREAD_POOL);

List<Callable<String>> callables = Arrays.asList(new DelayedCallable("fast thread", 100),new DelayedCallable("slow thread", 3000));

for (Callable<String> callable : callables) {
      service.submit(callable);
}

可以使用take()方法访问结果:


long startProcessingTime = System.currentTimeMillis();

Future<String> future = service.take();

String firstThreadResponse = future.get();

long totalProcessingTime = System.currentTimeMillis() - startProcessingTime;

assertTrue("First response should be from the fast thread","fast thread".equals(firstThreadResponse));

assertTrue(totalProcessingTime >= 100 && totalProcessingTime < 1000);

LOG.debug("Thread finished after: " + totalProcessingTime+ " milliseconds");

future = service.take();

String secondThreadResponse = future.get();

totalProcessingTime= System.currentTimeMillis() - startProcessingTime;

assertTrue(
"Last response should be from the slow thread","slow thread".equals(secondThreadResponse));

assertTrue(totalProcessingTime >= 3000 && totalProcessingTime < 4000);

LOG.debug("Thread finished after: " + totalProcessingTime+ " milliseconds");

awaitTerminationAfterShutdown(WORKER_THREAD_POOL);

6.结论

根据用例,我们有各种选项来等待线程完成执行。

一个CountDownLatch当我们需要一种机制来通知一组由其他线程执行的操作已完成一个或多个线程是非常有用的。

当我们需要尽快访问任务结果时,ExecutorCompletionService非常有用,当我们想要等待所有正在运行的任务完成时,其他方法。


关注公众号:「Java知己」,每天更新Java知识哦,期待你的到来!

  • 发送「1024」,免费领取 30 本经典编程书籍。
  • 发送「Group」,与 10 万程序员一起进步。
  • 发送「JavaEE实战」,领取《JavaEE实战》系列视频教程。
  • 发送「玩转算法」,领取《玩转算法》系列视频教程。
    image
    image
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ExecutorService等待线程完成后优雅结束 的相关文章

  • VSCode 中 js 文件类型注释报错的问题解决

    在阅读 Vue js 源码的时候遇见的一个问题 xff0c 本来可以忽略的玩意 xff0c 但是报错的波浪线 xff0c 如鲠在喉实在受不了 xff0c 解决之 报错说明 Type annotations can only be used
  • 3-3-07-nuxtjs案例realworld-nuxtjs

    realworld nuxtjs 项目地址 xff1a https gitee com dingxd9702 realworld nuxtjs 创建项目 mkdir realworld nuxtjsyarn init yyarn add n
  • 3-4-01-搭建自己的 Server Side Render

    搭建自己的 Server Side Render Vue 实例的服务端渲染 使用 vue server renderer 插件完成 vue 实例的服务端渲染使用 express 创建一个 node 服务器fs 模块读取 html 模板 sp
  • css实现超出部分显示省略号

    显示一行 xff0c 省略号 white span class token operator span space span class token punctuation span nowrap span class token punc
  • 正则表达式 匹配字符串 matches()方法的运用

    package cn niit demo5zhengze 正则表达式 匹配字符串 matches 方法的运用 public class Test public static void main String args checkQQ che
  • Blazor组件自做三 : 使用JS隔离封装ZXing扫码

    本文基础步骤参考前两篇文章 Blazor组件自做一 使用JS隔离封装viewerjs库 Blazor组件自做二 使用JS隔离制作手写签名组件 1 在文件夹wwwroot lib 添加zxing子文件夹 里面下载库文件 文件文末源码里可复制
  • zabbix报警方式,邮件报警和微信报警。

    整理csdn时候发现了一篇2016年整理zabbix报警不知道当时啥情况没有发布出去 xff0c 凑个数重新发布 最近这些天都在弄Zabbix不再只是简单的监控物理硬件 xff0c 服务端口 xff0c 流量图等 让Zabbix的功能发挥到
  • itk registration 1

    参考 xff1a https itk org ITKSoftwareGuide html Book2 ITKSoftwareGuide Book2ch3 html x26 1740003 18 图像配准是确定将一幅图像上的点映射到另一幅图像
  • 凤凰涅槃

    涅槃 xff0c 涅槃 xff0c 太阳般灿烂 xff0c 黑色的凤凰飞入烈火 xff0c 一切腐朽烧干 涅槃 xff0c 涅槃 xff0c 大海般坦然 xff0c 红色的凤凰浴火重生 xff0c 一切希望复燃 涅槃 xff0c 涅槃 xf
  • Qt单个模块的编译

    有可能使用的Qt没有自己需要的模块 xff0c 这个时候就只能自己去编译这个模 本次就拿QWebEngine这个模块来讲 xff0c 这个模块比较复杂 xff0c 编译很容易出问题
  • H5/js/web lottie解析json 播放视频或动画

    H5 js web lottie解析json 播放视频或动画 1 需要UI设计的小伙伴用AE把效果图导出时导成JSON格式 xff0c 并且把json引入到你需要用到的项目里面2 在index html中引入lottie min js3 查
  • VirtualBox虚拟机闪退后如何重新打开

    电脑 xff08 Mac系统 xff09 由于某种问题自动重启了 xff0c 重启后之前在用的VirtualBox xff08 虚拟机管理器 xff09 能恢复运行 xff0c 但VirtualBoxVM xff08 虚拟机终端 xff09
  • (五)建筑物多边形化简系列——最小外接矩形的获取

    最小外接矩形问题是在给出一个多边形 xff08 或一群点 xff09 xff0c 求出面积最小且外接多边形的矩形的问题 这个问题看起来并不难 xff0c 但是具体实现并不简单 除了调用现有的公开库之外 xff0c 这里给出一种简单且易理解的
  • 使用 LxRunOffline 实现 WSL 自定义安装、备份

    本文初衷是想在非默认目录下安装WSL xff0c 查阅资料后用LxRunOffline了解并实现了WSL自定义安装 前言 虽说目前的 WSL 在 Windows 10 生态中已经越发成熟 xff0c 但在实际使用中依旧存在一些不足之处 xf
  • [Python基础] @statismethod和@classmethod中的注意事项

    声明 xff1a 以下讨论将区别方法 xff08 method xff09 和函数 xff08 function xff09 两个概念 xff0c 方法依托于对象 xff0c 而函数可以脱离对象之外存在 众所周知 xff0c python类
  • centos安装chrome 无法运行 Running as root without --no-sandbox is not supported ygote_host_impl_linux.cc

    launcher Failed to get the debug url 0313 145713 562233 ERROR zygote host impl linux cc 90 Running as root without no sa
  • Blazor组件自做七: 使用JS隔离制作定位/持续定位组件

    1 运行截图 演示地址 2 在文件夹wwwroot lib 添加geolocation子文件夹 添加geolocation js文件 本组件主要是调用浏览器两个API实现基于浏览器的定位功能 现代桌面和移动端都支持 包括MAUI Blazo
  • 普通表转换分区表操作步骤

    普通表转换分区表操作步骤 1 转分区表原因 生产数据库 xff0c 一张表 xff0c 一亿多行数据 xff0c 绝大部分查询按月为维度做时间范围查询 xff0c 未分区状态下 xff0c 查询IO量大 xff0c 计划以分区截剪的方式减少
  • 信号量的实现和应用

    信号量的实现和应用 一 实验环境 本次实验的操作环境还是一样的实验环境 环境文件如下 xff1a 如果不清楚的话请参考往期博客 二 实验目标与内容 1 目标 xff1a 加深对进程同步与互斥概念的认识 xff1b 掌握信号量的使用 xff0
  • Linux常用软件包(常用命令)

    文章目录 Linux应用程序基础RPM包管理工具rpm命令的格式查询RPM软件包信息应用程序与系统命令的关系软件包的封装类型rpm软件包查询未安装的 RPM 软件包文件中信息安装 升级 卸载 RPM 软件包yum常用的操作命令 编译安装编译

随机推荐

  • KBQA的工作流程(基于检索的方法)

    KBQA的工作流程 基于检索的方法 图谱构建阶段 注 采用neo4j的方法进行建立图数据库 1 数据准备 结构化数据 例如mysql的关系型数据库 进行整理 导出为csv 半结构化数据 采用包装器技术 进行抽取 之后 进行清洗 文本数据 采
  • 可解压rar、zip且可打包7z的powershell命令

    安装模块 Install Module 7zip4powershell 为什么要使用7zip 1个原因是传统zip只支持最大2gb的文件 xff0c 而7zip支持 16 xff0c 000 xff0c 000TB大的文档 xff0c 另一
  • ubuntu18.04 systemd配置学习手册(1) --相关概念

    1 从init系统说起 Linux 操作系统的启动首先从 BIOS 开始 xff0c 接下来进入 boot loader xff0c 由 bootloader 载入内核 xff0c 进行内核初始化 内核初始化的最后一步就是启动 PID 为
  • ubuntu18.04 systemd配置学习手册(2) --具体使用

    转载 xff1a http www ruanyifeng com blog 2016 03 systemd tutorial part two html 1 开机启动 对于那些支持 Systemd 的软件 xff0c 安装的时候 xff0c
  • 建造者模式(Builder模式)详解

    在软件开发过程中有时需要创建一个复杂的对象 xff0c 这个复杂对象通常由多个子部件按一定的步骤组合而成 例如 xff0c 计算机是由 OPU 主板 内存 硬盘 显卡 机箱 显示器 键盘 鼠标等部件组装而成的 xff0c 采购员不可能自己去
  • 关于读取文件失败

    太久没遇上这个问题了 今天碰上刚好回顾一下 文件路径以及python语法中涉及到的反斜杠 转义符问题 xff0c 在python字符串中 有转义的含义 xff0c 如 t可代表TAB xff0c n代表换行 xff0c 所以我们需要采取一些
  • GKI改造原则、机制和方法

    Google在android11 5 4分支上开始要求所有下游厂商使用Generic Kernel Image xff08 GKI xff09 xff0c 需要将SoC和device相关的代码从核心内核剥离到可加载模块中 xff08 下文称
  • Blazor入门100天 : 身份验证和授权 (1) - 建立带身份验证工程

    目录 建立默认带身份验证 Blazor 程序角色 组件 特性 过程逻辑DB 改 Sqlite将自定义字段添加到用户表脚手架拉取IDS文件 本地化资源freesql 生成实体类 freesql 管理ids数据表初始化 Roles freesq
  • 黑屏定屏那些事 - 系统机制,分析套路和实战(系统篇)

    本文发表于内核工匠公众号 xff0c 旨在给内核开发的小伙伴分享 xff1a Android系统层面用户UI交互的设计 xff0c 从而理解手机黑屏定屏时背后的故事 Android系统对黑屏定屏类问题的维测思路 xff0c 有那些先进的思想
  • ArrayList和JSONArray边遍历边删除到底该怎么做

    话题看起来有点老生了 xff0c 但是加入JSONArray还是有很多新意 ArrayList 方式1 xff1a 索引遍历 List lt Integer gt list 61 new ArrayList lt gt list add 1
  • git 添加源地址和查看源地址

    git init 添加本地文件关联 git add commit 提交 git commit m 34 first commit 34 关联仓库 git remote add origin http git sfc com yuxang w
  • Python爬取淘宝商品数据,价值千元的爬虫外包项目

    前言 本文的文字及图片来源于网络 仅供学习 交流使用 不具有任何商业用途 如有问题请及时联系我们以作处理 PS xff1a 如有需要Python学习资料的小伙伴可以加点击下方链接自行获取 完整代码可以点击下方链接获取 python免费学习资
  • AndroidStudio安装kotlin插件

    转载请注明出处 xff1a http blog csdn net feibendexiaoma article details 72625846 前言 2017 Google I O大会宣布将Kotlin语言作为安卓开发的一级编程语言 xf
  • Ubuntu20.04/Ubuntu22.04 配置VScode+Opencv+cmake(C++)

    Ubuntu20 04 Ubuntu22 04 配置VScode 43 Opencv 43 cmake xff08 C 43 43 xff09 下面介绍Ubuntu20 04下安装opencv xff0c 当然Ubuntu22 04也适用
  • Ubuntu-解决包依赖关系

    Ubuntu 解决包依赖关系的办法 安装软件包的时候 xff0c 有时会遇到类似下图的依赖问题 xff0c 无法正常安装 xff0c 下面提供三种方法解决依赖问题 1 可以尝试用下面方法处理依赖问题 xff0c 紧跟前一条安装命令后面输入下
  • Ubuntu 18.04系统进不去了,只有老内核的恢复模式还能进,请大家帮看一下

    Ubuntu 18 04 新内核和恢复模式进不去 xff0c 会报错 老内核也进不去 xff0c 只有老内核的恢复模式能进 xff0c 请大佬帮看一下 virtualbox里安装的是Ubuntu 18 04 1 LTS xff0c 打开id
  • Android Lottie动画的简单使用

    简介 在Android中做动画效果无非是以下几种方法 xff1a 普通动画帧动画属性动画通过改变LayoutParams布局参数来实现动画 现如今在Github上有一个比较火的动画库Lottie xff0c Github上关于Lottie库
  • java运行提示没有 org/slf4j/LoggerFactory 或者 log4j

    问题描述 Exception in thread main java lang NoClassDefFoundError org slf4j LoggerFactory 解决办法 在pom xml里面添加对应的配置 将jar包大包进去 lt
  • Blazor入门100天 : 身份验证和授权 (3) - DB改Sqlite

    目录 建立默认带身份验证 Blazor 程序角色 组件 特性 过程逻辑DB 改 Sqlite将自定义字段添加到用户表脚手架拉取IDS文件 本地化资源freesql 生成实体类 freesql 管理ids数据表初始化 Roles freesq
  • ExecutorService等待线程完成后优雅结束

    1 概述 该ExecutorService框架可以很容易地在处理多线程任务 我们将举例说明我们等待线程完成执行的一些场景 此外 xff0c 我们将展示如何正常关闭ExecutorService并等待已经运行的线程完成其执行 2 Execut