0

0

Java并发编程中的ExecutorService应用

P粉602998670

P粉602998670

发布时间:2026-01-10 14:17:27

|

384人浏览过

|

来源于php中文网

原创

Future.get()卡住的根本原因是任务未结束或被I/O、锁等阻塞;其为同步阻塞调用,不主动中断任务;cancel(true)仅靠任务响应中断信号才生效,否则无效。

java并发编程中的executorservice应用

ExecutorService.submit() 返回的 Future 为什么有时 get() 会卡住

根本原因是任务本身没结束,或者被阻塞在 I/O、锁、无限循环里,Future.get() 是同步阻塞调用,它不会主动中断正在运行的任务。

  • 默认情况下,ExecutorService 不会因 Future.cancel(true) 就强制终止线程——JVM 不支持真正“杀掉”线程,只能靠任务自己响应 Thread.interrupted()
  • 如果任务中用了 Object.wait()Thread.sleep() 或可中断的 NIO 操作,cancel(true) 才会生效并抛出 InterruptedException
  • 常见陷阱:提交了一个无超时的 get(),而任务因数据库连接超时未设 timeout,结果主线程永久挂起

稳妥做法是始终带超时:

try {
    result = future.get(3, TimeUnit.SECONDS);
} catch (TimeoutException e) {
    future.cancel(true); // 尝试中断
    throw new RuntimeException("Task timed out", e);
}

shutdown() 和 shutdownNow() 的实际行为差异

shutdown() 是温柔退场:拒绝新任务,但允许已提交的(包括队列里排队的)任务继续执行完;shutdownNow() 是硬中断:尝试中断所有正在执行的任务,并清空任务队列,返回尚未执行的任务列表。

  • shutdownNow() 并不保证任务立刻停止——它只是对每个工作线程调用 Thread.interrupt(),能否响应取决于任务代码是否检查中断状态
  • 如果任务正持有 synchronized 锁或在 native 方法中,中断信号会被忽略,线程仍会继续执行直到自然退出
  • 生产环境建议优先用 shutdown() + awaitTermination() 组合,留出合理等待窗口

典型安全关闭流程:

立即学习Java免费学习笔记(深入)”;

AI Web Designer
AI Web Designer

AI网页设计师,快速生成个性化的网站设计

下载
executor.shutdown();
try {
    if (!executor.awaitTermination(10, TimeUnit.SECONDS)) {
        executor.shutdownNow();
        if (!executor.awaitTermination(5, TimeUnit.SECONDS)) {
            System.err.println("Pool did not terminate gracefully");
        }
    }
} catch (InterruptedException e) {
    executor.shutdownNow();
    Thread.currentThread().interrupt();
}

FixedThreadPool 为什么不适合处理大量短时 IO 任务

因为它的核心线程数固定且永不回收,一旦任务涉及网络/磁盘 I/O,线程会长时间阻塞在 read()accept() 等系统调用上,导致线程资源被无效占用,吞吐量骤降。

  • FixedThreadPool 底层用的是 LinkedBlockingQueue(无界队列),任务持续涌入时,队列无限增长,可能引发 OOM
  • 替代方案应选 newCachedThreadPool()(适合大量短生命周期任务)或更可控的 newThreadPoolExecutor() 配合 SynchronousQueue 和自定义拒绝策略
  • 现代推荐:直接用 ForkJoinPool.commonPool() 处理 CPU 密集型任务;IO 密集型则交给 Netty、Vert.x 等异步框架,而非裸 ExecutorService

使用 CompletionService 统一收集异步结果的必要场景

当多个异步任务耗时差异大,又需要“谁先完成就先处理谁的结果”,而不是按提交顺序 get(),就必须用 CompletionService。否则你得维护一堆 Future,轮询或按序阻塞,效率低且易出错。

  • CompletionService 内部封装了阻塞队列(如 LinkedBlockingQueue<future></future>),任务完成即入队,take() 总能拿到最先结束的那个
  • 注意:它不改变任务执行逻辑,只是结果获取方式更灵活;底层仍依赖传入的 ExecutorService
  • 别误以为它能“自动重试失败任务”——异常仍需在 future.get() 里显式捕获

简单用法示例:

ExecutorService executor = Executors.newFixedThreadPool(4);
CompletionService<String> cs = new ExecutorCompletionService<>(executor);
<p>for (int i = 0; i < 5; i++) {
cs.submit(() -> {
Thread.sleep(1000 * i); // 模拟不同耗时
return "result-" + i;
});
}</p><p>for (int i = 0; i < 5; i++) {
try {
String result = cs.take().get(); // 按完成顺序获取
System.out.println(result);
} catch (ExecutionException e) {
e.printStackTrace();
}
}

线程池不是万能胶,用错类型或忽略生命周期管理,比不用还危险。尤其要注意中断语义的脆弱性——它只是一次通知,不是指令。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

99

2025.12.01

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

389

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2111

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

357

2023.08.31

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 81.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号