FutureTask通常如何使用?

FutureTask 是一种有用的工具,用于管理并发编程中可取消的、可监听的任务。它通常被用来包装 CallableRunnable 对象,并在单独的线程中执行这些任务。以下是一些使用 FutureTask 的典型场景和步骤:

1. 包装 CallableRunnable

下面是基本的包装 CallableRunnable 的方法:

java 复制代码
// 使用 Callable
Callable<Integer> callable = () -> {
    // 模拟长时间的任务
    Thread.sleep(1000);
    return 123;
};

FutureTask<Integer> futureTask = new FutureTask<>(callable);

// 使用 Runnable
Runnable runnable = () -> {
    // 模拟长时间的任务
    try {
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        Thread.currentThread().interrupt();
    }
};
FutureTask<Void> futureTask2 = new FutureTask<>(runnable, null);

2. 提交任务

可以通过 ThreadExecutorService 提交 FutureTask 并执行任务:

通过 Thread
java 复制代码
Thread thread = new Thread(futureTask);
thread.start();
通过 ExecutorService
java 复制代码
ExecutorService executor = Executors.newSingleThreadExecutor();
executor.submit(futureTask);
executor.shutdown();

3. 获取结果

使用 FutureTaskget 方法可以获取任务的执行结果:

java 复制代码
try {
    Integer result = futureTask.get();
    System.out.println("Result: " + result);
} catch (InterruptedException | ExecutionException e) {
    e.printStackTrace();
}

如果需要带超时限制的获取,可以使用带超时参数的 get 方法:

java 复制代码
try {
    Integer result = futureTask.get(500, TimeUnit.MILLISECONDS);
    System.out.println("Result with timeout: " + result);
} catch (TimeoutException e) {
    System.out.println("Task timed out");
} catch (InterruptedException | ExecutionException e) {
    e.printStackTrace();
}

4. 取消任务

可以使用 cancel 方法取消任务,如果任务正在运行,你可以选择是否允许中断:

java 复制代码
boolean mayInterruptIfRunning = true;
futureTask.cancel(mayInterruptIfRunning);

if (futureTask.isCancelled()) {
    System.out.println("Task was cancelled");
} else {
    System.out.println("Task was not cancelled");
}

5. 完整示例

下面是一个完整的示例,展示如何使用 FutureTask 包装一个 Callable 任务,并通过 ExecutorService 提交任务执行,同时获取结果并处理取消:

java 复制代码
import java.util.concurrent.*;

public class FutureTaskExample {
    public static void main(String[] args) {
        // 创建一个 Callable 对象
        Callable<Integer> callable = () -> {
            System.out.println("Task started");
            Thread.sleep(2000); // 模拟长时间任务
            System.out.println("Task completed");
            return 123;
        };

        // 创建 FutureTask
        FutureTask<Integer> futureTask = new FutureTask<>(callable);

        // 使用 ExecutorService 提交 FutureTask
        ExecutorService executor = Executors.newSingleThreadExecutor();
        executor.submit(futureTask);

        // 主线程可以执行其他任务
        System.out.println("Main thread is doing something else");

        try {
            // 获取任务的结果
            Integer result = futureTask.get(1, TimeUnit.SECONDS);
            System.out.println("Result from FutureTask: " + result);
        } catch (InterruptedException e) {
            System.out.println("Task was interrupted");
        } catch (ExecutionException e) {
            System.out.println("Task encountered an execution exception");
        } catch (TimeoutException e) {
            System.out.println("Task timed out");
            futureTask.cancel(true); // 取消任务
            System.out.println("Task was cancelled: " + futureTask.isCancelled());
        } finally {
            // 关闭 Executor 服务
            executor.shutdown();
        }
    }
}

在这个示例中,FutureTask 包装一个 Callable 对象,并由 ExecutorService 执行。主线程可以执行其他任务,并通过 futureTask.get() 方法获取任务结果。超时发生时任务会被取消。

相关推荐
XH华1 小时前
C语言第十一章内存在数据中的存储
c语言·开发语言
叫我阿柒啊3 小时前
Java全栈开发面试实战:从基础到微服务架构
java·vue.js·spring boot·redis·git·full stack·interview
小凡敲代码3 小时前
2025年金九银十Java面试场景题大全:高频考点+深度解析+实战方案
java·程序员·java面试·后端开发·求职面试·java场景题·金九银十
AndrewHZ3 小时前
【python与生活】如何用Python写一个简单的自动整理文件的脚本?
开发语言·python·生活·脚本·文件整理
拉法豆粉3 小时前
在压力测试中如何确定合适的并发用户数?
java·开发语言
枯萎穿心攻击4 小时前
Unity VS UE 性能工具与内存管理
开发语言·游戏·unity·ue5·游戏引擎·虚幻·虚幻引擎
爱上纯净的蓝天4 小时前
迁移面试题
java·网络·c++·pdf·c#
老赵的博客4 小时前
c++ 常用接口设计
开发语言·c++
binbinaijishu884 小时前
Python爬虫入门指南:从零开始的网络数据获取之旅
开发语言·爬虫·python·其他
chenglin0164 小时前
Logstash_Input插件
java·开发语言