首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
如何在 Java 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果

如何在 Java 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果

热心网友
33
转载
2026-04-30

如何在 Ja va 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果

如何在 Ja va 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

处理异步任务时,你是否遇到过这样的困扰:提交了一堆任务,却只能按照提交顺序一个个等待结果,即便后面的任务先完成了也得干等着?这在处理网络请求或I/O操作时尤其低效。好在Ja va并发包里藏着一个利器——ExecutorCompletionService。它能帮你按任务实际完成的先后顺序获取结果,真正做到“谁先做完,谁先汇报”。

简单来说,它是对普通Executor的增强包装,核心秘诀在于内部用了一个BlockingQueue(默认是LinkedBlockingQueue)来暂存已完成的Future。这样一来,获取结果的顺序就与提交或执行的顺序脱钩了。

ExecutorCompletionService 按任务实际完成先后顺序返回结果,其核心是用BlockingQueue暂存已完成Future,通过done回调自动入队,take()/poll()按完成顺序出队获取Future再调用get()得结果。

核心原理:完成即入队,取即出队

它的工作机制非常清晰,就像一个高效的流水线分拣系统:

  • 当你通过submit()提交一个Callable任务后,ExecutorCompletionService会将其交给底层的Executor去执行。
  • 与此同时,它会创建一个封装该任务的Future,并巧妙地注册一个内部监听机制(这通常依赖于FutureTaskdone状态回调)。
  • 一旦任务执行完毕(无论是成功返回还是抛出异常),对应的Future就会被自动放入内部的阻塞队列中等待被领取。
  • 这时,你调用take()poll()方法,就能按照任务完成的先后顺序,从队列中取出这些Future,之后再调用get()便能拿到最终的结果或异常。

整个过程,实现了从“任务完成”到“结果可被消费”的无缝、有序衔接。

基本使用步骤(带示例)

光说不练假把式。假设我们有3个模拟的异步任务,它们的耗时各不相同(以此来模拟现实中网络或I/O操作的差异),我们的目标就是谁能干完活,就先处理谁的结果。

(以下为关键代码逻辑,可直接运行)

  • 第一步:准备线程池。 例如,使用Executors.newFixedThreadPool(3)创建一个固定大小的线程池。
  • 第二步:包装成CompletionService。 用这个线程池作为参数,构造一个ExecutorCompletionService实例。
  • 第三步:提交任务。 循环提交你的Callable任务(比如,每个任务休眠随机时间后返回一条耗时信息)。
  • 第四步:按完成顺序获取。 关键来了!调用completionService.take()。这个方法会阻塞当前线程,直到有任何一个任务完成,然后立刻返回那个最先完成的Future
  • 第五步:提取结果。 对返回的Future调用get()。由于这个Future对应的任务已经确定完成,此时调用get()会立即返回结果,不会发生阻塞。

通过这五步,你就能轻松实现“结果先到先得”,极大提升了处理效率。

处理异常与超时的注意事项

现实世界不会总是一帆风顺,任务可能失败,我们也不想无限期等待。使用ExecutorCompletionService时,有几点需要特别留意:

立即学习“Ja va免费学习笔记(深入)”;

  • 异常处理仍在。CompletionService取出的Future,调用其get()方法时,仍然可能抛出ExecutionException(它包装了任务执行时抛出的原始异常)或InterruptedException,因此必要的try-catch不可少。
  • 非阻塞与超时选项。 如果不想让take()一直阻塞,可以使用poll()方法(没有已完成任务则立即返回null),或者使用poll(long timeout, TimeUnit unit)来设置一个等待超时时间。
  • 分清Future和结果。 务必记住:take()poll()返回的是Future对象,并不是最终的结果值。你必须再调用一次get(),才能拿到实际的计算结果或感知到执行异常。
  • 失败任务的处理。 如果某个任务执行失败,而你想忽略它继续处理其他已完成的任务,可以在调用get()时捕获ExecutionException,然后根据业务逻辑决定是跳过、记录还是重试。一个任务的失败不会阻塞队列,其他已完成的任务依然可以被正常取出。

对比 Future 列表 + 循环 isDone() 的劣势

在没有CompletionService之前,一个常见的替代方案是:将提交任务后得到的所有Future保存到一个列表里,然后写一个循环去不停地轮询每个FutureisDone()方法。

这种方法存在几个明显的短板:效率低下(轮询本身消耗CPU)、响应不实时(从任务完成到被轮询发现存在延迟)、容易写出忙等待(busy-wait)代码,并且线程安全性需要额外小心。

反观ExecutorCompletionService,它基于阻塞队列的“生产者-消费者”模型,天然实现了“完成即通知”的机制。消费线程在无任务时安心阻塞,有任务时立刻被唤醒,没有任何空转开销。同时,其内部封装保证了线程安全,使用语义也清晰直观——这就是专业工具带来的降维打击。

来源:https://www.php.cn/faq/2393354.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

如何在 Java 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果
编程语言
如何在 Java 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果

如何在 Ja va 中使用 ExecutorCompletionService 按照异步任务完成的先后顺序获取返回结果 处理异步任务时,你是否遇到过这样的困扰:提交了一堆任务,却只能按照提交顺序一个个等待结果,即便后面的任务先完成了也得干等着?这在处理网络请求或I O操作时尤其低效。好在Ja va并

热心网友
04.30
怎么利用 java.util.Arrays.mismatch() 快速找出两个配置数组中第一个不一致的配置项
编程语言
怎么利用 java.util.Arrays.mismatch() 快速找出两个配置数组中第一个不一致的配置项

如何用 Arrays mismatch() 快速定位配置数组的首个差异项 在配置比对或数据校验的场景里,你是不是也写过循环来逐项比较两个数组?其实,直接用 Arrays mismatch() 就能一步到位,精准锁定第一个差异点的索引。这个方法简直就是为“找不同”量身定制的,不仅代码更简洁,还内置了空

热心网友
04.30
怎么在 Java 中声明并初始化基础数据类型(int, double, boolean)
编程语言
怎么在 Java 中声明并初始化基础数据类型(int, double, boolean)

怎么在 Ja va 中声明并初始化基础数据类型(int, double, boolean) 声明并初始化 int 变量时,别漏掉分号和类型关键字 Ja va 的强类型特性,意味着每个变量都必须有明确的“身份”。int 就是 int,不能像 Ja vaScript 那样用一个 let 或 var 就糊

热心网友
04.30
如何在 Java 中使用 AtomicInteger 实现无锁的线程安全计数
编程语言
如何在 Java 中使用 AtomicInteger 实现无锁的线程安全计数

如何在 Ja va 中使用 AtomicInteger 实现无锁的线程安全计数 先来看一个核心的技术论断:AtomicInteger的incrementAndGet通常比synchronized快,因为它基于CPU的CAS指令,避免了阻塞和上下文切换的开销。但事情总有另一面:在高争用场景下,它可能因

热心网友
04.30
如何在 Java 中通过 Constructor.newInstance() 动态创建类的实例对象
编程语言
如何在 Java 中通过 Constructor.newInstance() 动态创建类的实例对象

Constructor newInstance()已过时,应改用getDeclaredConstructor() setAccessible(true) newInstance()或Unsafe allocateInstance();它抛出InvocationTargetException是为包装构

热心网友
04.30

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

Mac如何使用BetterTouchTool增强触控_Mac BetterTouchTool增强触控步骤
系统平台
Mac如何使用BetterTouchTool增强触控_Mac BetterTouchTool增强触控步骤

一、授予系统权限并启动基础服务 想让BetterTouchTool真正“活”起来,第一步就得打通系统权限。它需要“辅助功能”权限来监听你的触控板事件,也需要“屏幕录制”权限来执行一些窗口操作。这两项权限缺一不可,否则你会发现手势做了,但电脑毫无反应。 具体操作其实不复杂:先进入系统「设置」-「隐私与

热心网友
04.30
如何开启Windows 11“高性能模式” 解决笔记本玩游戏掉帧降频方法
系统平台
如何开启Windows 11“高性能模式” 解决笔记本玩游戏掉帧降频方法

如何开启Windows 11“高性能模式” 解决笔记本玩游戏掉帧降频方法 笔记本玩游戏,最扫兴的莫过于画面突然卡顿、帧率断崖式下跌。很多时候,问题并非出在硬件本身,而是Windows 11默认的电源策略在“拖后腿”。为了省电,系统会动态调节处理器频率、让核心休眠,甚至给显卡设置功耗墙,这直接限制了硬

热心网友
04.30
Mac系统更新失败提示错误的解决方法
系统平台
Mac系统更新失败提示错误的解决方法

macOS更新失败?别慌,这五步能帮你搞定 升级macOS时,进度条卡住不动、弹窗提示“无法验证更新”或者干脆报错退出,这事儿确实让人头疼。其实,这些看似随机的故障,背后通常逃不出几个核心原因:存储空间不连续、网络连接不干净、缓存文件有冲突,或者磁盘底层出了点小状况。别担心,按照下面这套经过验证的步

热心网友
04.30
Linux下使用Jattach工具诊断Java进程 零停机获取Dump信息
系统平台
Linux下使用Jattach工具诊断Java进程 零停机获取Dump信息

Linux下使用Jattach工具诊断Ja va进程 零停机获取Dump信息 开门见山,先说一个核心判断:jattach 并非 JDK 自带工具,也不能直接替代 jstack。但它的价值在于,能在某些棘手场景下,绕过 JVM 的安全限制成功获取 dump。当然,这有个前提——目标 JVM 的 Att

热心网友
04.30
Linux怎么安装和配置Tyk API网关 Linux开源网关管理详解
系统平台
Linux怎么安装和配置Tyk API网关 Linux开源网关管理详解

Tyk Dashboard 启动失败?从配置到排查的完整指南 在Linux上部署Tyk,可不是简单的apt install或yum install就能搞定。它背后依赖着MongoDB和Redis,并且对配置顺序有严格的要求。跳过其中任何一环,tyk-dashboard服务很可能就会卡在502错误,或

热心网友
04.30