Java使用Callable和Future获取线程结果详解
时间:2026-02-02 13:00:58 342浏览 收藏
“纵有疾风来,人生不言弃”,这句话送给正在学习文章的朋友们,也希望在阅读本文《Java如何用Callable和Future获取线程结果》后,能够真的帮助到大家。我也会在后续的文章中,陆续更新文章相关的技术文章,有好的建议欢迎大家在评论留言,非常感谢!
Callable 和 Runnable 的核心区别在于:Callable 的 call() 方法有返回值且可抛异常,Runnable 的 run() 方法无返回值且不能抛受检异常;前者适用于需获取结果的场景,后者适用于无需返回的场景。

Callable 和 Runnable 的核心区别在哪
Runnable 的 run() 方法不能返回值、也不能抛出受检异常;Callable 的 call() 方法必须返回一个泛型结果,且可以抛出异常。这意味着如果你需要线程执行后拿到计算结果(比如查询数据库返回 List),就必须用 Callable,而不是 Runnable。
Runnable适合“只做事不回话”的场景,例如日志写入、缓存刷新Callable适合“做完要交作业”的场景,例如远程接口调用、复杂计算、文件解析- 直接 new Thread(new Callable(...)) 是**行不通的**——
Thread构造器只接受Runnable,必须通过ExecutorService提交
如何用 ExecutorService 提交 Callable 并获取 Future
Future 是对异步计算结果的“占位符”,它不保存结果本身,而是提供检查、等待、取值的控制能力。关键点是:Future 不会自动触发执行,必须由线程池调度。
ExecutorService executor = Executors.newFixedThreadPool(2);
Callable<String> task = () -> {
Thread.sleep(1000);
return "done";
};
Future<String> future = executor.submit(task); // 提交后立即返回 Future
// 后续可多次调用,但 get() 是阻塞的
try {
String result = future.get(); // 等待完成并取值
} catch (ExecutionException e) {
// 注意:原始异常被包装在 ExecutionException 中,e.getCause() 才是你的业务异常
} finally {
executor.shutdown();
}
future.isDone()判断是否执行完毕(非阻塞)future.cancel(true)尝试中断正在运行的任务(是否生效取决于任务内部是否响应中断)future.get(3, TimeUnit.SECONDS)带超时的取值,超时抛出TimeoutException
Future.get() 阻塞问题怎么破
直接调用 future.get() 会卡住当前线程,这在 Web 请求或 UI 线程中不可接受。常见解法不是“避免 Future”,而是“换一种等法”。
- 用
isDone()+ 自旋轮询(仅限低频、短耗时任务,否则浪费 CPU) - 用
get(timeout, unit)控制最大等待时间,超时后走降级逻辑(如返回缓存值) - 升级到
CompletableFuture:支持回调(thenApply)、组合(thenCombine)、异常处理(exceptionally),真正实现非阻塞链式异步 - 注意:多个
Future.get()串行调用仍是同步等待,想并行取结果,得用invokeAll()或CompletableFuture.allOf()
为什么有时 Future.get() 拿不到预期异常
当你在 call() 中 throw new IOException(),future.get() 却抛出 ExecutionException,而不是原始异常——这是设计使然,Future 统一将任务内抛出的任何异常包装为 ExecutionException 的 cause。
try {
future.get();
} catch (ExecutionException e) {
Throwable cause = e.getCause();
if (cause instanceof IOException) {
// 这里才能捕获你 throw 的 IOException
}
}
- 不要直接 catch
IOException,那是徒劳的 - 如果任务因线程被中断而失败,
e.getCause()可能是CancellationException - 如果任务还没开始就被 cancel,
future.isCancelled()为 true,此时调用get()也会抛CancellationException
Future 当作普通对象反复调用 get(),又没设超时,很容易让整个调用链卡死。到这里,我们也就讲完了《Java使用Callable和Future获取线程结果详解》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
378 收藏
-
419 收藏
-
337 收藏
-
149 收藏
-
226 收藏
-
177 收藏
-
308 收藏
-
384 收藏
-
173 收藏
-
258 收藏
-
454 收藏
-
163 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习