Future

Future是Java5增加的类,它用来描述一个异步计算的结果。你可以使用 isDone 方法检查计算是否完成,或者使用 get 方法阻塞住调用线程,直到计算完成返回结果。你也可以使用 cancel 方法停止任务的执行。下面来一个栗子:

  1. public class FutureDemo {
  2.  
  3. public static void main(String[] args) {
  4. ExecutorService es = Executors.newFixedThreadPool(10);
  5. Future<Integer> f = es.submit(() ->{
  6. Thread.sleep(10000);
  7. // 结果
  8. return 100;
  9. });
  10.  
  11. // do something
  12.  
  13. Integer result = f.get();
  14. System.out.println(result);
  15.  
  16. // while (f.isDone()) {
  17. // System.out.println(result);
  18. // }
  19. }
  20. }

在这个例子中,我们往线程池中提交了一个任务并立即返回了一个Future对象,接着可以做一些其他操作,最后利用它的 get 方法阻塞等待结果或 isDone 方法轮询等待结果(关于Future的原理可以参考之前的文章:【并发编程】Future模式及JDK中的实现

虽然这些方法提供了异步执行任务的能力,但是对于结果的获取却还是很不方便,只能通过阻塞或者轮询的方式得到任务的结果。

阻塞的方式显然和我们的异步编程的初衷相违背,轮询的方式又会耗费无谓的CPU资源,而且也不能及时的得到计算结果,为什么不能用观察者设计模式当计算结果完成及时通知监听者呢?

很多语言,比如Node.js,采用Callback的方式实现异步编程。Java的一些框架,比如Netty,自己扩展了Java的 Future 接口,提供了 addListener 等多个扩展方法。Google的guava也提供了通用的扩展Future:ListenableFuture 、 SettableFuture 以及辅助类 Futures 等,方便异步编程。为此,Java终于在JDK1.8这个版本中增加了一个能力更强的Future类:CompletableFuture 。它提供了非常强大的Future的扩展功能,可以帮助我们简化异步编程的复杂性,提供了函数式编程的能力,可以通过回调的方式处理计算结果。下面来看看这几种方式。

Netty-Future

引入Maven依赖:

  1. <dependency>
  2. <groupId>io.netty</groupId>
  3. <artifactId>netty-all</artifactId>
  4. <version>4.1.29.Final</version>
  5. </dependency>
  1. public class NettyFutureDemo {
  2.  
  3. public static void main(String[] args) throws InterruptedException {
  4. EventExecutorGroup group = new DefaultEventExecutorGroup(4);
  5. System.out.println("开始:" + DateUtils.getNow());
  6.  
  7. Future<Integer> f = group.submit(new Callable<Integer>() {
  8. @Override
  9. public Integer call() throws Exception {
  10. System.out.println("开始耗时计算:" + DateUtils.getNow());
  11. Thread.sleep(10000);
  12. System.out.println("结束耗时计算:" + DateUtils.getNow());
  13. return 100;
  14. }
  15. });
  16.  
  17. f.addListener(new FutureListener<Object>() {
  18. @Override
  19. public void operationComplete(Future<Object> objectFuture) throws Exception {
  20. System.out.println("计算结果:" + objectFuture.get());
  21. }
  22. });
  23.  
  24. System.out.println("结束:" + DateUtils.getNow());
  25. // 不让守护线程退出
  26. new CountDownLatch(1).await();
  27. }
  28. }

输出结果:

  1. 开始:2019-05-16 08:25:40:779
  2. 结束:2019-05-16 08:25:40:788
  3. 开始耗时计算:2019-05-16 08:25:40:788
  4. 结束耗时计算:2019-05-16 08:25:50:789
  5. 计算结果:100

从结果可以看出,耗时计算结束后自动触发Listener的完成方法,避免了主线程无谓的阻塞等待,那么它究竟是怎么做到的呢?下面看源码

DefaultEventExecutorGroup 实现了 EventExecutorGroup 接口,而 EventExecutorGroup 则是实现了JDK ScheduledExecutorService 接口的线程组接口,所以它拥有线程池的所有方法。然而它却把所有返回 java.util.concurrent.Future 的方法重写为返回 io.netty.util.concurrent.Future ,把所有返回 java.util.concurrent.ScheduledFuture 的方法重写为返回 io.netty.util.concurrent.ScheduledFuture 。

  1. public interface EventExecutorGroup extends ScheduledExecutorService, Iterable<EventExecutor> {
  2. /**
  3. * 返回一个EventExecutor
  4. */
  5. EventExecutor next();
  6.  
  7. Iterator<EventExecutor> iterator();
  8.  
  9. Future<?> submit(Runnable task);
  10. <T> Future<T> submit(Runnable task, T result);
  11. <T> Future<T> submit(Callable<T> task);
  12.  
  13. ScheduledFuture<?> schedule(Runnable command, long delay, TimeUnit unit);
  14. <V> ScheduledFuture<V> schedule(Callable<V> callable, long delay, TimeUnit unit);
  15. ScheduledFuture<?> scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit);
  16. ScheduledFuture<?> scheduleWithFixedDelay(Runnable command, long initialDelay, long delay, TimeUnit unit);
  17. }

EventExecutorGroup 的submit方法因为 newTaskFor 的重写导致返回了netty的 Future 实现类,而这个实现类正是 PromiseTask 。

  1. @Override
  2. public <T> Future<T> submit(Callable<T> task) {
  3. return (Future<T>) super.submit(task);
  4. }
  5.  
  6. @Override
  7. protected final <T> RunnableFuture<T> newTaskFor(Callable<T> callable) {
  8. return new PromiseTask<T>(this, callable);
  9. }

PromiseTask 的实现很简单,它缓存了要执行的 Callable 任务,并在run方法中完成了任务调用和Listener的通知。

  1. @Override
  2. public void run() {
  3. try {
  4. if (setUncancellableInternal()) {
  5. V result = task.call();
  6. setSuccessInternal(result);
  7. }
  8. } catch (Throwable e) {
  9. setFailureInternal(e);
  10. }
  11. }
  12.  
  13. @Override
  14. public Promise<V> setSuccess(V result) {
  15. if (setSuccess0(result)) {
  16. notifyListeners();
  17. return this;
  18. }
  19. throw new IllegalStateException("complete already: " + this);
  20. }
  21.  
  22. @Override
  23. public Promise<V> setFailure(Throwable cause) {
  24. if (setFailure0(cause)) {
  25. notifyListeners();
  26. return this;
  27. }
  28. throw new IllegalStateException("complete already: " + this, cause);
  29. }

任务调用成功或者失败都会调用 notifyListeners 来通知Listener,所以大家得在回调的函数里调用 isSuccess 方法来检查状态。

这里有一个疑惑,会不会 Future 在调用 addListener 方法的时候任务已经执行完成了,这样子会不会通知就会失败了啊?

  1. @Override
  2. public Promise<V> addListener(GenericFutureListener<? extends Future<? super V>> listener) {
  3. synchronized (this) {
  4. addListener0(listener);
  5. }
  6.  
  7. if (isDone()) {
  8. notifyListeners();
  9. }
  10.  
  11. return this;
  12. }

可以发现,在Listener添加成功之后,会立即检查状态,如果任务已经完成立刻进行回调,所以这里不用担心啦。OK,下面看看Guava-Future的实现。

Guava-Future

首先引入guava的Maven依赖:

  1. <dependency>
  2. <groupId>com.google.guava</groupId>
  3. <artifactId>guava</artifactId>
  4. <version>22.0</version>
  5. </dependency>
  1. public class GuavaFutureDemo {
  2.  
  3. public static void main(String[] args) throws InterruptedException {
  4. System.out.println("开始:" + DateUtils.getNow());
  5.  
  6. ExecutorService executorService = Executors.newFixedThreadPool(10);
  7. ListeningExecutorService service = MoreExecutors.listeningDecorator(executorService);
  8. ListenableFuture<Integer> future = service.submit(new Callable<Integer>() {
  9. @Override
  10. public Integer call() throws Exception {
  11. System.out.println("开始耗时计算:" + DateUtils.getNow());
  12. Thread.sleep(10000);
  13. System.out.println("结束耗时计算:" + DateUtils.getNow());
  14. return 100;
  15. }
  16. });
  17.  
  18. future.addListener(new Runnable() {
  19. @Override
  20. public void run() {
  21. System.out.println("调用成功");
  22. }
  23. }, executorService);
  24. System.out.println("结束:" + DateUtils.getNow());
  25. new CountDownLatch(1).await();
  26. }
  27. }

ListenableFuture 可以通过 addListener 方法增加回调函数,一般用于不在乎执行结果的地方。如果需要在执行成功时获取结果或者执行失败时获取异常信息,需要用到 Futures 工具类的 addCallback 方法:

  1. Futures.addCallback(future, new FutureCallback<Integer>() {
  2. @Override
  3. public void onSuccess(@Nullable Integer result) {
  4. System.out.println("成功,计算结果:" + result);
  5. }
  6.  
  7. @Override
  8. public void onFailure(Throwable t) {
  9. System.out.println("失败");
  10. }
  11. }, executorService);

前面提到除了 ListenableFuture 外,还有一个 SettableFuture 类也支持回调能力。它实现自 ListenableFuture ,所以拥有 ListenableFuture 的所有能力。

  1. public class GuavaFutureDemo {
  2.  
  3. public static void main(String[] args) throws InterruptedException {
  4. System.out.println("开始:" + DateUtils.getNow());
  5. ExecutorService executorService = Executors.newFixedThreadPool(10);
  6. ListenableFuture<Integer> future = submit(executorService);
  7. Futures.addCallback(future, new FutureCallback<Integer>() {
  8. @Override
  9. public void onSuccess(@Nullable Integer result) {
  10. System.out.println("成功,计算结果:" + result);
  11. }
  12.  
  13. @Override
  14. public void onFailure(Throwable t) {
  15. System.out.println("失败:" + t.getMessage());
  16. }
  17. }, executorService);
  18. Thread.sleep(1000);
  19. System.out.println("结束:" + DateUtils.getNow());
  20. new CountDownLatch(1).await();
  21. }
  22.  
  23. private static ListenableFuture<Integer> submit(Executor executor) {
  24. SettableFuture<Integer> future = SettableFuture.create();
  25. executor.execute(new Runnable() {
  26. @Override
  27. public void run() {
  28. System.out.println("开始耗时计算:" + DateUtils.getNow());
  29. try {
  30. Thread.sleep(3000);
  31. } catch (InterruptedException e) {
  32. e.printStackTrace();
  33. }
  34. System.out.println("结束耗时计算:" + DateUtils.getNow());
  35. // 返回值
  36. future.set(100);
  37. // 设置异常信息
  38. // future.setException(new RuntimeException("custom error!"));
  39. }
  40. });
  41. return future;
  42. }
  43. }

看起来用法上没有太多差别,但是有一个很容易被忽略的重要问题。当 SettableFuture 的这种方式最后调用了 cancel 方法后,线程池中的任务还是会继续执行,而通过 submit 方法返回的 ListenableFuture 方法则会立即取消执行,这点尤其要注意。下面看看源码:

和Netty的Future一样,Guava也是通过实现了自定义的 ExecutorService 实现类 ListeningExecutorService 来重写了 submit 方法。

  1. public interface ListeningExecutorService extends ExecutorService {
  2. <T> ListenableFuture<T> submit(Callable<T> task);
  3. ListenableFuture<?> submit(Runnable task);
  4. <T> ListenableFuture<T> submit(Runnable task, T result);
  5. }

同样的,newTaskFor 方法也被进行了重写,返回了自定义的Future类:TrustedListenableFutureTask

  1. @Override
  2. protected final <T> RunnableFuture<T> newTaskFor(Runnable runnable, T value) {
  3. return TrustedListenableFutureTask.create(runnable, value);
  4. }
  5.  
  6. @Override
  7. protected final <T> RunnableFuture<T> newTaskFor(Callable<T> callable) {
  8. return TrustedListenableFutureTask.create(callable);
  9. }

任务调用会走 TrustedFutureInterruptibleTask 的run方法:

  1. @Override
  2. public void run() {
  3. TrustedFutureInterruptibleTask localTask = task;
  4. if (localTask != null) {
  5. localTask.run();
  6. }
  7. }
  8.  
  9. @Override
  10. public final void run() {
  11. if (!ATOMIC_HELPER.compareAndSetRunner(this, null, Thread.currentThread())) {
  12. return; // someone else has run or is running.
  13. }
  14. try {
  15. // 抽象方法,子类进行重写
  16. runInterruptibly();
  17. } finally {
  18. if (wasInterrupted()) {
  19. while (!doneInterrupting) {
  20. Thread.yield();
  21. }
  22. }
  23. }
  24. }

最终还是调用到 TrustedFutureInterruptibleTask 的 runInterruptibly 方法,等待任务完成后调用 set 方法。

  1. @Override
  2. void runInterruptibly() {
  3. if (!isDone()) {
  4. try {
  5. set(callable.call());
  6. } catch (Throwable t) {
  7. setException(t);
  8. }
  9. }
  10. }
  11.  
  12. protected boolean set(@Nullable V value) {
  13. Object valueToSet = value == null ? NULL : value;
  14. // CAS设置值
  15. if (ATOMIC_HELPER.casValue(this, null, valueToSet)) {
  16. complete(this);
  17. return true;
  18. }
  19. return false;
  20. }

在 complete 方法的最后会获取到Listener进行回调。

上面提到的 SettableFuture 和 ListenableFuture 的 cancel 方法效果不同,原因在于一个重写了 afterDone 方法而一个没有。

下面是 ListenableFuture 的 afterDone 方法:

  1. @Override
  2. protected void afterDone() {
  3. super.afterDone();
  4.  
  5. if (wasInterrupted()) {
  6. TrustedFutureInterruptibleTask localTask = task;
  7. if (localTask != null) {
  8. localTask.interruptTask();
  9. }
  10. }
  11.  
  12. this.task = null;
  13. }

wasInterrupted 用来判断是否调用了 cancel (cancel方法会设置一个取消对象Cancellation到value中)

  1. protected final boolean wasInterrupted() {
  2. final Object localValue = value;
  3. return (localValue instanceof Cancellation) && ((Cancellation) localValue).wasInterrupted;
  4. }

interruptTask 方法通过线程的 interrupt 方法真正取消线程任务的执行:

  1. final void interruptTask() {
  2. Thread currentRunner = runner;
  3. if (currentRunner != null) {
  4. currentRunner.interrupt();
  5. }
  6. doneInterrupting = true;
  7. }

由 Callback Hell 引出 Promise 模式

如果你对 ES6 有所接触,就不会对 Promise 这个模式感到陌生,如果你对前端不熟悉,也不要紧,我们先来看看回调地狱(Callback Hell)是个什么概念。

回调是一种我们推崇的异步调用方式,但也会遇到问题,也就是回调的嵌套。当需要多个异步回调一起书写时,就会出现下面的代码(以 js 为例):

  1. asyncFunc1(opt, (...args1) => {
  2. asyncFunc2(opt, (...args2) => {
  3. asyncFunc3(opt, (...args3) => {
  4. asyncFunc4(opt, (...args4) => {
  5. // some operation
  6. });
  7. });
  8. });
  9. });

虽然在 JAVA 业务代码中很少出现回调的多层嵌套,但总归是个问题,这样的代码不易读,嵌套太深修改也麻烦。于是 ES6 提出了 Promise 模式来解决回调地狱的问题。可能就会有人想问:java 中存在 Promise 模式吗?答案是肯定的。

前面提到了 Netty 和 Guava 的扩展都提供了 addListener 这样的接口,用于处理 Callback 调用,但其实 jdk1.8 已经提供了一种更为高级的回调方式:CompletableFuture。首先尝试用 CompletableFuture 来重写上面回调的问题。

  1. public class CompletableFutureTest {
  2.  
  3. public static void main(String[] args) throws InterruptedException {
  4. System.out.println("开始:" + DateUtils.getNow());
  5. CompletableFuture<Integer> completableFuture = CompletableFuture.supplyAsync(() -> {
  6. System.out.println("开始耗时计算:" + DateUtils.getNow());
  7. try {
  8. Thread.sleep(10000);
  9. } catch (InterruptedException e) {
  10. e.printStackTrace();
  11. }
  12. System.out.println("结束耗时计算:" + DateUtils.getNow());
  13. return 100;
  14. });
  15. completableFuture.whenComplete((result, e) -> {
  16. System.out.println("回调结果:" + result);
  17. });
  18. System.out.println("结束:" + DateUtils.getNow());
  19. new CountDownLatch(1).await();
  20. }
  21. }

使用CompletableFuture耗时操作没有占用主线程的时间片,达到了异步调用的效果。我们也不需要引入任何第三方的依赖,这都是依赖于 java.util.concurrent.CompletableFuture 的出现。CompletableFuture 提供了近 50 多个方法,大大便捷了 java 多线程操作,和异步调用的写法。

使用 CompletableFuture 解决回调地狱问题:

  1. public class CompletableFutureDemo {
  2. public static void main(String[] args) throws InterruptedException {
  3. long l = System.currentTimeMillis();
  4. CompletableFuture<Integer> completableFuture = CompletableFuture.supplyAsync(() -> {
  5. System.out.println("在回调中执行耗时操作...");
  6. Thread.sleep(10000);
  7. return 100;
  8. });
  9. completableFuture = completableFuture.thenCompose(i -> {
  10. return CompletableFuture.supplyAsync(() -> {
  11. System.out.println("在回调的回调中执行耗时操作...");
  12. Thread.sleep(10000);
  13. return i + 100;
  14. });
  15. });
  16. completableFuture.whenComplete((result, e) -> {
  17. System.out.println("计算结果:" + result);
  18. });
  19. System.out.println("主线程运算耗时:" + (System.currentTimeMillis() - l) + " ms");
  20. new CountDownLatch(1).await();
  21. }
  22. }

输出:

  1. 在回调中执行耗时操作...主线程运算耗时:58 ms在回调的回调中执行耗时操作...计算结果:200

使用 thenCompose 或者 thenComposeAsync 等方法可以实现回调的回调,且写出来的方法易于维护。

总的看来,为Future模式增加回调功能就不需要阻塞等待结果的返回并且不需要消耗无谓的CPU资源去轮询处理状态,JDK8之前使用Netty或者Guava提供的工具类,JDK8之后则可以使用自带的 CompletableFuture 类。Future 有两种模式:将来式和回调式。而回调式会出现回调地狱的问题,由此衍生出了 Promise 模式来解决这个问题。这才是 Future 模式和 Promise 模式的相关性。

【并发编程】Future模式添加Callback及Promise 模式的更多相关文章

  1. 并发编程学习笔记(9)----AQS的共享模式源码分析及CountDownLatch使用及原理

    1. AQS共享模式 前面已经说过了AQS的原理及独享模式的源码分析,今天就来学习共享模式下的AQS的几个接口的源码. 首先还是从顶级接口acquireShared()方法入手: public fin ...

  2. Java多线程编程模式实战指南之Promise模式

    Promise模式简介(转) Promise模式是一种异步编程模式 .它使得我们可以先开始一个任务的执行,并得到一个用于获取该任务执行结果的凭据对象,而不必等待该任务执行完毕就可以继续执行其他操作.等 ...

  3. 并发编程-Future+callable+FutureTask 闭锁机制

    项目中经常有些任务需要异步(提交到线程池中)去执行,而主线程往往需要知道异步执行产生的结果,这时我们要怎么做呢?用runnable是无法实现的,我们需要用callable实现. FutureTask ...

  4. Java并发编程-再谈 AbstractQueuedSynchronizer 1 :独占模式

    关于AbstractQueuedSynchronizer JDK1.5之后引入了并发包java.util.concurrent,大大提高了Java程序的并发性能.关于java.util.concurr ...

  5. Java并发编程:Future接口、FutureTask类

    在前面的文章中我们讲述了创建线程的2种方式,一种是直接继承Thread,另外一种就是实现Runnable接口. 这2种方式都有一个缺陷就是:在执行完任务之后无法获取执行结果. 如果需要获取执行结果,就 ...

  6. 并发编程 05—— Callable和Future

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

  7. 并发编程 01—— ThreadLocal

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

  8. 并发编程 20—— AbstractQueuedSynchronizer 深入分析

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

  9. 并发编程 02—— ConcurrentHashMap

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

随机推荐

  1. 用python实现的抓取腾讯视频所有电影的爬虫

    1. [代码]用python实现的抓取腾讯视频所有电影的爬虫    # -*- coding: utf-8 -*-# by awakenjoys. my site: www.dianying.atim ...

  2. SSH三大框架的搭建整合(struts2+spring+hibernate)(转)

    原文地址:http://blog.csdn.net/kyle0349/article/details/51751913  尊重原创,请访问原文地址 SSH说的上是javaweb经典框架,不能说100% ...

  3. codeforces 633D D. Fibonacci-ish(dfs+暴力+map)

    D. Fibonacci-ish time limit per test 3 seconds memory limit per test 512 megabytes input standard in ...

  4. codeforces 633A A. Ebony and Ivory(暴力)

    A. Ebony and Ivory time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  5. 通过rtmpdump推送海康视频流到red5服务器

    现在主流的网络摄像机都支持标准H264视频格式,例如 海康网络摄像机, 通过海康提供的网络SDK可以获取到视频码流.我测试的这款相机,视频编码采用的是H264,音频编码采用的是G711a. 这里,我仅 ...

  6. Guice总结

    Guice总结 Jar包:guice-4.1.0.jar 辅包: guava-15.0.jar aopalliance-.jar javaee-api-6.0-RC2.jar Guice的IoC 两种 ...

  7. bzoj 2093 [Poi2010]Frog——滑动窗口

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2093 找第k近的可以用一个含k个元素的滑动窗口来实现. 卡空间也还行,但卡时间.不要预处理倍 ...

  8. jquery给select赋值

    项目中用到通过ajax请求数据然后给select赋值,由于经常遇到类似的代码,在这里把整个过程记录一下. 首选发出ajax请求如下: <script type="text/javasc ...

  9. 人物-IT-雷军:雷军

    ylbtech-人物-IT-雷军:雷军 雷军 (全国工商联副主席,小米科技创始人.董事长) 雷军,1969年12月16日出生于湖北仙桃,毕业于武汉大学,是中国大陆著名天使投资人.  雷军作为中国互联网 ...

  10. Zend Server 安装与配置图文教程

    Zend Server是一款专业的PHP Web开发应用服务器,一些初次接触并使用此程序的朋友可能不太了解安装方法,本文为您提供了Zend Server 安装与配置图文教程,欢迎大家阅读,并提出自己的 ...