news 2026/4/29 0:35:05

聊聊 Future 接口

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
聊聊 Future 接口

一、背景介绍

在前几篇线程系列文章中,我们介绍了线程池的相关技术,任务执行类只需要实现Runnable接口,然后交给线程池,就可以轻松的实现异步执行多个任务的目标,提升程序的执行效率,比如如下异步执行任务下载。

// 创建一个线程池 ExecutorService executor = Executors.newFixedThreadPool(2); // 提交任务 executor.submit(new Runnable() { @Override public void run() { // 执行下载某文件任务 System.out.println("执行下载某文件任务"); } });

而实际上Runnable接口并不能满足所有的需求,比如有些场景下,我们想要获取任务执行的返回结果,Runnable接口因为无返回值,只能想办法通过额外的方式来写入和读取,操作起来十分不便。

因此,从 JDK 1.5 开始,Java 标准库提供了一个Callable接口,与Runnable接口相比,它的方法上多了一个返回值;同时Callable是一个泛型接口,可以返回指定类型的结果,比如如下的实现类!

public class Task implements Callable<String> { @Override public String call() throws Exception { // 执行下载某文件任务 System.out.println("执行下载某文件任务"); return "xxx"; } }

问题来了,如何获取异步执行的结果呢?

在 JDK 1.5 中,Java 标准库还提供了一个Future接口,它可以用来获取异步执行的结果。

下面我们一起来了解一下这个Future接口!

二、Future

Future接口,表示一个可能还没有完成异步任务的结果,它提供了检查任务是否已完成、以及等待任务完成并获取结果等方法。

如果看过ExecutorService.submit()方法,会发现它的返回参数都是Future类型,Future类型的实例可以用来获取异步任务执行的结果。

下面我们先来看一个简单的示例,以便于更好的理解!

public class Task implements Callable<String> { @Override public String call() throws Exception { // 执行下载某文件任务,并返回文件名称 System.out.println("thread name:" + Thread.currentThread().getName() + " 开始执行下载任务"); return "xxx.png"; } }
public class FutureTest { public static void main(String[] args) throws Exception { // 创建一个线程池 ExecutorService executor = Executors.newFixedThreadPool(1); // 初始化一个任务 Callable<String> task = new Task(); // 提交任务并获得Future的实例 Future<String> future = executor.submit(task); // 从Future获取异步执行返回的结果(可能会阻塞等待结果) String result =future.get(); System.out.println("任务执行结果:" + result); // 任务执行完毕之后,关闭线程池(可选) executor.shutdown(); } }

输出结果如下:

thread name:pool-1-thread-1 开始执行下载任务 任务执行结果:xxx.png

从以上的示例可以清晰的看到,当需要获取异步线程的执行结果返回值时,通常需要搭配使用FutureCallable接口来实现,大体可以用如下步骤来概括:

  • 1.首先提交一个实现Callable接口的任务到线程池中

  • 2.然后获取一个Future类型的对象

  • 3.最后在主线程中调用Future对象的get()方法,如果异步任务执行完成,就可以直接获得结果;如果异步任务执行没有完成,get()方法会阻塞,直到任务执行完成后才能获取结果

分析源码你会发现,Callable接口主要用途是定义一个支持返回结果的方法;重点实现主要集中在Future接口上。

下面我们重点来看下Future接口方法!

2.1、Future 接口方法

方法

描述

get()

获取结果(会阻塞等待)

get(long timeout, TimeUnit unit)

在指定的时间内获取结果,如果超时,会抛异常并退出等待状态

cancel(boolean mayInterruptIfRunning)

尝试取消当前任务,当传入参数为true时,表示尝试中断任务的执行,false表示不中断,继续执行直到完成,如果取消成功,返回true;反之false

isCancelled()

判断任务是否已取消

isDone()

判断任务是否已完成

2.2、Future 接口实现类

Future本质其实是一个接口,并不是具体的实现类,真正负责工作的还是它的实现类来完成。

我们还是以上文的线程池ExecutorService.submit()方法为例,看看它用的是哪种实现类!

分析一下源码,会发现线程池用的实现类是FutureTask,关键核心源码如下:

protected <T> RunnableFuture<T> newTaskFor(Callable<T> callable) { return new FutureTask<T>(callable); }

FutureTask类是一个实现了Future接口所有功能的具体类,可直接使用它来实现获取异步任务执行的结果值。

FutureTask的工作原理其实也并不复杂,它接受一个Callable或者Runnable对象作为参数,然后在线程池执行器中执行该任务,最后通过get()方法可以同步等待获取任务的执行结果。

真正起到关键作用的是,在FutureTask内部,封装了一个状态变量,用于记录任务的状态(等待、运行、完成、取消等),以及任务执行结果或异常信息,通过该状态变量,我们可以判断任务是否已完成、以及获取任务的执行结果等信息。

因为FutureTask也实现了Runnable接口,因此我们也可以将FutureTask作为任务,提交给线程池执行器。

具体示例如下:

public class FutureTest { public static void main(String[] args) throws Exception { // 1.创建一个线程池 ExecutorService executor = Executors.newFixedThreadPool(1); // 2.初始化一个任务 Callable<String> callable = new Task(); // 3.创建FutureTask对象 FutureTask<String> futureTask = new FutureTask<>(callable); // 4.提交任务给执行器执行 executor.execute(futureTask); // 5.获取任务的执行结果 String result = futureTask.get(3, TimeUnit.SECONDS); System.out.println("任务执行结果:" + result); // 6.关闭线程池(可选) executor.shutdown(); } }

输出结果同上!

如果想尝试取消任务的执行,也可以通过如下方式来实现!

boolean isSuccess = futureTask.cancel(true); System.out.println("任务是否取消成功:" + isSuccess);

除此之外,如果仔细的分析Future接口的类关系,会发现它的实现类非常的多,FutureTask只是它的一个基础实现类而已,部分类关系图如下!

其它常用实现类简介:

  • CompletableFuture:支持传入回调对象,当异步任务完成或者发生异常时,自动调用回调对象的回调方法

  • ForkJoinTask:支持把一个大任务拆成多个小任务,然后并行执行,在多核 CPU 上可以显著提升程序的执行效率

  • ScheduledFuture:支持周期性定时的执行任务,其中ScheduledFutureTask是一个私有类,只能通过ScheduledThreadPoolExecutor初始化操作

关于CompletableFutureForkJoinTaskScheduledFuture,我们会在后面的文章中,再次单独介绍具体的用法。

三、小结

本文主要围绕Future接口用法做了一次简单的知识总结,其中FutureTask类是Future接口中一个非常重要的实现类,通过它可以获取异步任务执行的返回值,通常用于异步计算带有返回值的任务。

限于篇幅的原因,本文没有对FutureTask做过深入的原理讲解,主要围绕具体用法进行介绍,有兴趣的朋友可以阅读这篇文章《Java的Future机制详解》,以便更清晰的了解它的实现原理。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/24 23:45:50

ResNet18微调实战:10元预算就能完成迁移学习

ResNet18微调实战&#xff1a;10元预算就能完成迁移学习 引言&#xff1a;小预算也能玩转AI模型定制 作为创业公司的技术负责人&#xff0c;你是否遇到过这样的困境&#xff1a;需要定制一个图像分类模型&#xff0c;但动辄上万的训练成本让人望而却步&#xff1f;今天我要分…

作者头像 李华
网站建设 2026/4/24 23:09:00

ResNet18半监督学习:小样本场景下的实战方案

ResNet18半监督学习&#xff1a;小样本场景下的实战方案 引言 在初创公司或小型团队中&#xff0c;我们常常面临数据量不足的困境。想象一下&#xff0c;你正在教小朋友认识动物&#xff0c;但手头只有几张猫狗的照片&#xff0c;这时候该怎么办&#xff1f;半监督学习就像一…

作者头像 李华
网站建设 2026/4/28 8:46:49

ResNet18模型轻量化指南:云端低成本完成模型压缩测试

ResNet18模型轻量化指南&#xff1a;云端低成本完成模型压缩测试 引言 作为一名边缘设备开发者&#xff0c;你是否经常遇到这样的困扰&#xff1a;想要优化ResNet18模型的参数量&#xff0c;但在本地测试不同剪枝方案时&#xff0c;每次训练都要耗费数小时甚至更长时间&#…

作者头像 李华
网站建设 2026/4/23 13:15:08

零代码玩转AI分类器:可视化界面+云端GPU,5分钟出结果

零代码玩转AI分类器&#xff1a;可视化界面云端GPU&#xff0c;5分钟出结果 1. 为什么你需要这个AI分类器&#xff1f; 作为市场专员&#xff0c;每天面对海量用户反馈时&#xff0c;你是否遇到过这些困扰&#xff1a; - 手工分类几百条用户留言要花大半天时间 - 想用AI工具但…

作者头像 李华
网站建设 2026/4/24 13:17:59

2025年度电商趋势报告:AI驱动、消费趋势与跨境机遇|附300+份报告PDF、数据、可视化模板汇总下载

原文链接&#xff1a;https://tecdat.cn/?p44767 原文出处&#xff1a;拓端抖音号拓端tecdat 引言 2025年&#xff0c;电商行业正经历一场由“流量争夺”向“价值重构”的深层变革。消费理性化与情绪需求的双重拉扯、AI技术对流量逻辑的颠覆、跨境业务的全球化布局与治理升级…

作者头像 李华
网站建设 2026/4/23 22:01:21

信息与网络安全核心速查手册:面试复习与工作自查必备基础知识集

一、概述 1.网络信息安全基本概念 信息安全&#xff1a;是指信息网络中的硬件、软件及其系统中的数据受到保护&#xff0c;不受偶然的或者恶意的原因而遭到破坏、更改、泄露、否认等&#xff0c;系统连续可靠正常的运行&#xff0c;信息服务不中断。 **密码学&#xff1a;**…

作者头像 李华