Future/FutureTask & Callable/Runnable & Executor/ExecutorService/Executors

本文详细介绍了Java并发编程中Callable和Future接口的使用方法,对比了Runnable接口的区别,并提供了三种不同的示例来展示如何使用FutureTask及ExecutorService进行异步计算。

用途:笔记。
各类具体方法参阅JDK API

Callable/Runnable接口

  • Runnable用于实现线程。大多数情况下,如果只想重写 run() 方法,而不重写其他 Thread 方法,那么应使用 Runnable 接口。这很重要,因为除非程序员打算修改或增强类的基本行为,否则不应为该类创建子类。
  • Callable 接口类似于 Runnable,两者都是为那些其实例可能被另一个线程执行的类设计的。但是 Runnable 不会返回结果,并且无法抛出经过检查的异常。

Note: Callable需要Override的方法是call(),带返回值,而Runnable是run(),不带返回值。

Future接口/FutureTask实现类

用于异步计算。

Future就是对于具体的Runnable或者Callable任务的执行结果进行取消、查询是否完成、获取结果、设置结果操作。get()方法会阻塞,直到任务返回结果。

Executor顶级接口/ExecutorService次级接口/Executors类

Executor就是Runnable和Callable的调度容器。可以执行Runnable/Callable对象:void execute(Runnable command)

示例

Demo1

  • 有实现Callable的对象,作为FutureTask的初始化参数;
  • 使用Thread启动FutureTask异步子线程;
  • 使用FutureTask的get()返回Callable对象异步执行的返回值;
  • 没Executor屁事儿。
import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

@SuppressWarnings("all")
public class FutureTaskDemo {
    public static void main(String[] args) {
        // 初始化一个Callable对象和FutureTask对象
        Callable pAccount = new PrivateAccount();
        FutureTask futureTask = new FutureTask(pAccount);
        // 使用futureTask创建一个线程
        Thread pAccountThread = new Thread(futureTask);
        System.out.println("futureTask线程现在开始启动,启动时间为:" + System.nanoTime());
        pAccountThread.start();
        System.out.println("主线程开始执行其他任务");
        // 从其他账户获取总金额
        int totalMoney = new Random().nextInt(100000);
        System.out.println("现在你在其他账户中的总金额为" + totalMoney);
        System.out.println("等待私有账户总金额统计完毕...");
        // 测试后台的计算线程是否完成,如果未完成则等待
        while (!futureTask.isDone()) {
            try {
                Thread.sleep(500);
                System.out.println("私有账户计算未完成继续等待...");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println("futureTask线程计算完毕,此时时间为" + System.nanoTime());
        Integer privateAccountMoney = null;
        try {
            privateAccountMoney = (Integer) futureTask.get();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("您现在的总金额为:" + totalMoney + privateAccountMoney.intValue());
    }
}

@SuppressWarnings("all")
class PrivateAccount implements Callable {
    Integer totalMoney;

    @Override
    public Object call() throws Exception {
        Thread.sleep(5000);
        totalMoney = new Integer(new Random().nextInt(10000));
        System.out.println("您当前有" + totalMoney + "在您的私有账户中");
        return totalMoney;
    }

}

Demo2


  • 使用匿名Callable对象作为FutureTask的初始化参数;
  • 不使用Thread,而使用ExecutorService(继承自父类接口Executor)的execute(Runnable)异步启动FutureTask子线程;
  • 使用FutureTask的get()返回Callable对象异步执行的返回值。

区别:引入Executor异步执行FutureTask对象,不再使用Thread异步执行Callable对象。
import java.util.Random;
import java.util.concurrent.*;

@SuppressWarnings("all")
public class FutureTaskDemoAndExecutor {
    public static void main(String[] args) {
        // 初始化一个Callable对象和FutureTask对象
//        Callable pAccount = new PrivateAccount();
        FutureTask futureTask = new FutureTask(new Callable<Integer>() {        // 用不用泛型皆可
            Integer totalMoney;
            @Override
            public Integer call() throws Exception {
                Thread.sleep(5000);
            totalMoney = new Integer(new Random().nextInt(10000));
            System.out.println("您当前有" + totalMoney + "在您的私有账户中");
            return totalMoney;
            }
        });
        // 使用futureTask创建一个线程
//        Thread pAccountThread = new Thread(futureTask);
        ExecutorService executorService = Executors.newSingleThreadExecutor();
        System.out.println("futureTask线程现在开始启动,启动时间为:" + System.nanoTime());
//        pAccountThread.start();
        executorService.execute(futureTask);
        System.out.println("主线程开始执行其他任务");
        // 从其他账户获取总金额
        int totalMoney = new Random().nextInt(100000);
        System.out.println("现在你在其他账户中的总金额为" + totalMoney);
        System.out.println("等待私有账户总金额统计完毕...");
        // 测试后台的计算线程是否完成,如果未完成则等待
        while (!futureTask.isDone()) {
            try {
                Thread.sleep(500);
                System.out.println("私有账户计算未完成继续等待...");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println("futureTask线程计算完毕,此时时间为" + System.nanoTime());
        Integer privateAccountMoney = null;
        try {
            privateAccountMoney = (Integer) futureTask.get();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("您现在的总金额为:" + totalMoney + privateAccountMoney.intValue());
    }
}

/*
@SuppressWarnings("all")
class PrivateAccount implements Callable {
    Integer totalMoney;

    @Override
    public Object call() throws Exception {
        Thread.sleep(5000);
        totalMoney = new Integer(new Random().nextInt(10000));
        System.out.println("您当前有" + totalMoney + "在您的私有账户中");
        return totalMoney;
    }

}*/

Demo3


  • 使用ExecutorService的submit(Callable<T>)直接异步执行匿名Callable对象
  • Future作为submit(Callable<T>)返回值的接收者;

区别:使用ExecutorService的 submit()而不是 execute(),省去了FutureTask,但是需要使用Future,调用 get()接收匿名Callable对象的返回值。
import java.util.Random;
import java.util.concurrent.*;

@SuppressWarnings("all")
public class FutureTaskDemoAndExecutorSubmit {
    public static void main(String[] args) {
        // 初始化一个Callable对象和FutureTask对象
//        FutureTask futureTask = new FutureTask(new Callable() {
//            Integer totalMoney;
//            @Override
//            public Object call() throws Exception {
//                Thread.sleep(5000);
//                totalMoney = new Integer(new Random().nextInt(10000));
//                System.out.println("您当前有" + totalMoney + "在您的私有账户中");
//                return totalMoney;
//            }
//        });
        // 使用futureTask创建一个线程
//        Thread pAccountThread = new Thread(futureTask);
        ExecutorService executorService = Executors.newSingleThreadExecutor();
        System.out.println("futureTask线程现在开始启动,启动时间为:" + System.nanoTime());
//        executorService.execute(futureTask);
        Future future = executorService.submit(new Callable<Integer>() {        // 用不用泛型皆可
            Integer totalMoney;
            @Override
            public Integer call() throws Exception {
                Thread.sleep(5000);
                totalMoney = new Integer(new Random().nextInt(10000));
                System.out.println("您当前有" + totalMoney + "在您的私有账户中");
                return totalMoney;
            }
        });
        System.out.println("主线程开始执行其他任务");
        // 从其他账户获取总金额
        int totalMoney = new Random().nextInt(100000);
        System.out.println("现在你在其他账户中的总金额为" + totalMoney);
        System.out.println("等待私有账户总金额统计完毕...");
        // 测试后台的计算线程是否完成,如果未完成则等待
//        while (!futureTask.isDone()) {
        while (!future.isDone()) {
            try {
                Thread.sleep(500);
                System.out.println("私有账户计算未完成继续等待...");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println("futureTask线程计算完毕,此时时间为" + System.nanoTime());
        Integer privateAccountMoney = null;
        try {
//            privateAccountMoney = (Integer) futureTask.get();
            privateAccountMoney = (Integer) future.get();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("您现在的总金额为:" + totalMoney + privateAccountMoney.intValue());
    }
}

匿名Callable泛型可用可不用。

### Java 多线程中的 CallableRunnableFutureFutureTask #### Callable 接口 `Callable&lt;V&gt;` 是一个类似于 `Runnable` 的接口,但是它可以返回结果并抛出异常。这使得它非常适合用于那些需要返回计算结果的任务。 ```java public interface Callable&lt;V&gt; { V call() throws Exception; } ``` 通过实现 `call()` 方法,可以在任务完成后获得返回值[^4]。 #### Runnable 接口 `Runnable` 是最常用的并发编程接口之一,通常用于定义不带返回值的任务。该接口只包含一个无参数的方法: ```java public interface Runnable { void run(); } ``` 由于 `run()` 不支持返回值也不允许声明受检异常,因此适用于不需要反馈结果的简单任务。 #### Future 接口 `Future&lt;V&gt;` 表示异步计算的结果。此接口提供了检查计算是否完成、等待其完成以及检索结果的方法。值得注意的是,在获取结果之前如果任务尚未结束,则会阻塞当前线程直到任务完成为止。 ```java public interface Future&lt;V&gt; { boolean cancel(boolean mayInterruptIfRunning); boolean isCancelled(); boolean isDone(); V get() throws InterruptedException, ExecutionException; V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException; } ``` 这些特性让开发者能够更好地管理长时间运行的操作而不会影响主线程的工作流程[^3]。 #### FutureTask 类 作为实现了 `RunnableFuture&lt;V&gt;` 接口的具体类,`FutureTask&lt;T&gt;` 可以包装 `Callable&lt;V&gt;` 或者 `Runnable` 对象来进行更复杂的操作处理。除了继承自父级的所有功能外,还增加了对状态变化的支持(比如设置已完成的状态),从而简化了多线程环境下的开发工作量。 ```java public class FutureTask&lt;V&gt; implements RunnableFuture&lt;V&gt; ``` 当传入的是 `Callable` 实例时可以直接调用它的 `get()` 来取得最终结果;如果是普通的 `Runnable` 则默认认为是没有输出值的情况[^5]。 ### 使用场景与代码实例 对于只需要执行某些动作而不关心它们产生的任何数据的情形下可以选择使用 `Runnable` 。然而当我们希望得到一些有用的信息或者是遇到可能失败的任务时就应该考虑采用 `Callable` 加上相应的错误捕捉机制来代替传统的做法。 下面给出几个简单的例子展示如何利用上述组件构建高效的并发应用程序: - **仅需执行任务** ```java // 创建一个新的线程并启动之 Thread thread = new Thread(new Task()); thread.start(); static class Task implements Runnable { @Override public void run() { System.out.println(&quot;Executing task...&quot;); } } ``` - **带有返回值的任务** ```java ExecutorService executor = Executors.newSingleThreadExecutor(); Future&lt;Integer&gt; future = executor.submit(() -&gt; { int sum = 0; for (int i=1;i&lt;=100;++i){ sum += i; } return sum; }); try{ Integer result = future.get(); // 阻塞直至任务完成 }catch(Exception e){ e.printStackTrace(); } executor.shutdownNow(); ``` - **组合使用 FutureTask** ```java ExecutorService pool = Executors.newFixedThreadPool(2); // 将 Callable 转换成 FutureTask FutureTask&lt;String&gt; ft1 = new FutureTask&lt;&gt;(new MyCallable()); pool.execute(ft1); // 提交到线程池中去跑 String res = null; if (!ft1.isDone()){ try { res = ft1.get(); // 获取结果前先判断是否已经完成了 } catch (InterruptedException | ExecutionException ex) { Logger.getLogger(Test.class.getName()).log(Level.SEVERE,null,ex); } } System.out.printf(&quot;Result from FutureTask: %s\n&quot;,res); pool.shutdown(); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值