• Java进阶篇--线程池之FutureTask


    目录

    FutureTask简介

    FutureTask的基本使用

    FutureTask的应用场景


    FutureTask简介

    FutureTask是Java中的一个类,用于表示可获取结果的异步任务。它实现了java.util.concurrent.Future接口,提供了启动和取消异步任务、查询任务是否已完成以及获取最终结果的方法。通过调用get()方法可以获取异步任务的结果,但会阻塞当前线程直至任务执行结束。一旦任务执行结束,任务不能重新启动或取消,除非调用runAndReset()方法。在FutureTask的源码中定义了多种状态,包括

    1. private static final int NEW = 0;
    2. private static final int COMPLETING = 1;
    3. private static final int NORMAL = 2;
    4. private static final int EXCEPTIONAL = 3;
    5. private static final int CANCELLED = 4;
    6. private static final int INTERRUPTING = 5;
    7. private static final int INTERRUPTED = 6;

    根据FutureTask.run()方法的执行的时机,FutureTask分为了3种状态:

    1、未启动状态:指的是在创建FutureTask后,但尚未执行FutureTask.run()方法之前,FutureTask处于未启动状态。

    2、已启动状态:指的是FutureTask.run()方法正在被执行的过程中,此时FutureTask处于已启动状态。

    3、已完成状态:包括了三种情况:

    • FutureTask.run()方法执行结束;
    • 调用FutureTask.cancel(…)方法取消任务;
    • 在执行任务期间抛出异常。

    以上三种情况都属于FutureTask的已完成状态。

    对于FutureTask的get()方法和cancel()方法,它们在不同状态下的行为总结如下:

    1、get方法:

    • 当FutureTask处于未启动或已启动状态时,执行FutureTask.get()方法将导致调用线程阻塞,直到任务执行完成并返回结果。
    • 如果FutureTask处于已完成状态,调用FutureTask.get()方法将立即返回结果或者抛出异常。

    2、cancel方法:

    • 当FutureTask处于未启动状态时,执行FutureTask.cancel()方法将任务标记为已取消,但不会影响当前未启动的任务。
    • 当FutureTask处于已启动状态时,执行FutureTask.cancel(true)方法将以中断线程的方式来阻止任务继续进行,如果执行FutureTask.cancel(false)将不会对正在执行任务的线程有任何影响,任务会继续执行直至结束。
    • 当FutureTask处于已完成状态时,执行FutureTask.cancel(…)方法将返回false,表示无法取消已经完成的任务。

    FutureTask的基本使用

    FutureTask除了实现了Future接口,还实现了Runnable接口,这使得它既可以作为一个异步任务,也可以作为一个可执行的任务被线程执行。因此,它可以交给Executor执行,也可以由调用的线程直接执行(FutureTask.run())。另外,通过ExecutorService的submit()方法也可以返回一个FutureTask对象,然后可以通过FutureTask.get()或者FutureTask.cancel方法获取结果或取消任务的执行。

    以下是一个简单的示例,演示了如何使用FutureTask来实现多线程任务的协调和管理,

    1. import java.util.concurrent.*;
    2. public class main {
    3. public static void main(String[] args) {
    4. // 创建一个FutureTask对象
    5. FutureTask futureTask = new FutureTask<>(() -> {
    6. // 模拟一个耗时的任务,这里使用Thread.sleep来模拟
    7. Thread.sleep(2000);
    8. return 42; // 返回结果
    9. });
    10. // 创建一个线程池
    11. ExecutorService executor = Executors.newFixedThreadPool(1);
    12. // 使用线程池执行FutureTask
    13. executor.execute(futureTask);
    14. // 在主线程中可以做一些其他的事情
    15. try {
    16. // 获取FutureTask的结果,如果任务还未完成,则会阻塞当前线程直到任务完成
    17. int result = futureTask.get();
    18. System.out.println("任务执行结果:" + result);
    19. } catch (InterruptedException | ExecutionException e) {
    20. e.printStackTrace();
    21. }
    22. // 关闭线程池
    23. executor.shutdown();
    24. }
    25. }

    在这个示例中,我们创建了一个FutureTask对象来执行一个耗时的任务,然后将它交给线程池来执行。在主线程中,我们可以做一些其他的事情,然后通过futureTask.get()方法来获取任务的执行结果。如果任务还未完成,get()方法会阻塞当前线程直到任务完成。

    这个示例展示了如何使用FutureTask来实现异步任务的执行和结果获取,同时也展示了使用ExecutorService来执行线程任务,并在任务完成后关闭线程池。

    FutureTask的应用场景

    在实际应用中,FutureTask通常用于以下场景:

    • 需要等待其他线程执行完毕后再继续执行:通过FutureTask可以将需要等待的任务封装起来,然后在需要等待的地方调用其get()方法,当前线程就会阻塞,直到任务执行完毕并返回结果。
    • 多个线程执行同一个任务的控制:可以使用FutureTask来保证多个线程对同一个任务的执行是有序的,例如只允许一个线程执行任务,其他线程需要等待任务执行完毕后才能继续。

    总的来说,FutureTask提供了一种方便的方式来管理异步任务的执行和获取结果,同时也可以用于多线程任务的协调与控制。当需要在多线程环境下进行任务调度和管理时,FutureTask是一个非常有用的工具。

  • 相关阅读:
    前端element表格导出excel
    PostgreSQL之SQL高级特性
    牛皮了!阿里面试官终于分享出了 2022 年最新的 java 面试题及答案
    Bootstrap的卡片组件相关知识
    python中的图像增强技术
    静态代码块
    Android Profiler入门与实践
    2021 第四届 浙江省大学生网络与信息安全竞赛技能赛 决赛 Writeup,5题
    【强化学习论文】离线元强化学习中基于对比学习的稳定表示
    华为校招第三题 找最小数
  • 原文地址:https://blog.csdn.net/m0_74293254/article/details/134290671