189 8069 5689

CompletableFuture真香,可以替代CountDownLatch!

[[420499]]

创新互联建站主营阿图什网站建设的网络公司,主营网站建设方案,成都app开发,阿图什h5小程序开发搭建,阿图什网站营销推广欢迎阿图什等地区企业咨询

在对类的命名这篇长文中,我们提到了Future和Promise。

Future相当于一个占位符,代表一个操作将来的结果。一般通过get可以直接阻塞得到结果,或者让它异步执行然后通过callback回调结果。

但如果回调中嵌入了回调呢?如果层次很深,就是回调地狱。Java中的CompletableFuture其实就是Promise,用来解决回调地狱问题。Promise是为了让代码变得优美而存在的。

有多优美?这么说吧,一旦你使用了CompletableFuture,就会爱不释手,就像初恋女友一样,天天想着她。

一系列静态方法

从它的源代码中,我们可以看到,CompletableFuture直接提供了几个便捷的静态方法入口。其中有run和supply两组。

run的参数是Runnable,而supply的参数是Supplier。前者没有返回值,而后者有,否则没有什么两样。

这两组静态函数,都提供了传入自定义线程池的功能。如果你用的不是外置的线程池,那么它就会使用默认的ForkJoin线程池。默认的线程池,大小和用途你是控制不了的,所以还是建议自己传递一个。

典型的代码,写起来是这个样子。

 
 
 
 
  1. CompletableFuture future = CompletableFuture.supplyAsync(()->{ 
  2.  return "test"; 
  3. }); 
  4. String result = future.join(); 

拿到CompletableFuture后,你就可以做更多的花样。

这些花样有很多

我们说面说了,CompletableFuture的主要作用,就是让代码写起来好看。配合Java8之后的stream流,可以把整个计算过程抽象成一个流。前面任务的计算结果,可以直接作为后面任务的输入,就像是管道一样。

 
 
 
 
  1. thenApply 
  2. thenApplyAsync 
  3. thenAccept 
  4. thenAcceptAsync 
  5. thenRun 
  6. thenRunAsync 
  7. thenCombine 
  8. thenCombineAsync 
  9. thenCompose 
  10. thenComposeAsync 

比如,下面代码的执行结果是99,并不因为是异步就打乱代码执行的顺序了。

 
 
 
 
  1. CompletableFuture cf = CompletableFuture.supplyAsync(() -> 10) 
  2.                 .thenApplyAsync((e) -> { 
  3.                     try { 
  4.                         Thread.sleep(10000); 
  5.                     } catch (InterruptedException ex) { 
  6.                         ex.printStackTrace(); 
  7.                     } 
  8.                     return e * 10; 
  9.                 }).thenApplyAsync(e -> e - 1); 
  10.  
  11. cf.join(); 
  12. System.out.println(cf.get()); 

同样的,函数的作用还要看then后面的动词。

  • apply 有入参和返回值,入参为前置任务的输出
  • accept 有入参无返回值,会返回CompletableFuture
  • run 没有入参也没有返回值,同样会返回CompletableFuture
  • combine 形成一个复合的结构,连接两个CompletableFuture,并将它们的2个输出结果,作为combine的输入
  • compose 将嵌套的CompletableFuture平铺开,用来串联两个CompletableFuture

when和handle

上面的函数列表,其实还有很多。比如:

 
 
 
 
  1. whenComplete 

when的意思,就是任务完成时候的回调。比如我们上面的例子,打算在完成任务后,输出一个done。它也是属于只有入参没有出参的范畴,适合放在最后一步进行观测。

 
 
 
 
  1. CompletableFuture cf = CompletableFuture.supplyAsync(() -> 10) 
  2.                 .thenApplyAsync((e) -> { 
  3.                     try { 
  4.                         Thread.sleep(1000); 
  5.                     } catch (InterruptedException ex) { 
  6.                         ex.printStackTrace(); 
  7.                     } 
  8.                     return e * 10; 
  9.                 }).thenApplyAsync(e -> e - 1) 
  10.                 .whenComplete((r, e)->{ 
  11.                     System.out.println("done"); 
  12.                 }) 
  13.                 ; 
  14.  
  15. cf.join(); 
  16. System.out.println(cf.get()); 

handle和exceptionally的作用,和whenComplete是非常像的。

 
 
 
 
  1. public CompletableFuture exceptionally(Function fn); 
  2.  
  3. public  CompletionStage handle(BiFunction fn); 

CompletableFuture的任务是串联的,如果它的其中某一步骤发生了异常,会影响后续代码的运行的。

exceptionally从名字就可以看出,是专门处理这种情况的。比如,我们强制某个步骤除以0,发生异常,捕获后返回-1,它将能够继续运行。

 
 
 
 
  1. CompletableFuture cf = CompletableFuture.supplyAsync(() -> 10) 
  2.                 .thenApplyAsync(e->e/0) 
  3.                 .thenApplyAsync(e -> e - 1) 
  4.                 .exceptionally(ex->{ 
  5.                     System.out.println(ex); 
  6.                     return -1; 
  7.                 }); 
  8.  
  9. cf.join(); 
  10. System.out.println(cf.get()); 

handle更加高级一些,因为它除了一个异常参数,还有一个正常的入参。处理方法也都类似,不再赘述。

当然,CompletableFuture的函数不仅仅这些,还有更多,根据函数名称很容易能够了解到它的作用。它还可以替换复杂的CountDownLatch,这要涉及到几个比较难搞的函数。

替代CountDownLatch

考虑下面一个场景。某一个业务接口,需要处理几百个请求,请求之后再把这些结果给汇总起来。

如果顺序执行的话,假设每个接口耗时100ms,那么100个接口,耗时就需要10秒。假如我们并行去获取的话,那么效率就会提高。

使用CountDownLatch可以解决。

 
 
 
 
  1. ExecutorService executor = Executors.newFixedThreadPool(5); 
  2.  
  3. CountDownLatch countDown = new CountDownLatch(requests.size()); 
  4. for(Request request:requests){ 
  5.     executor.execute(()->{ 
  6.         try{ 
  7.         //some opts 
  8.         }finally{ 
  9.             countDown.countDown(); 
  10.         } 
  11.     }); 
  12. countDown.await(200,TimeUnit.MILLISECONDS); 

我们使用CompletableFuture来替换它。

 
 
 
 
  1. ExecutorService executor = Executors.newFixedThreadPool(5); 
  2.  
  3. List> futureList = requests 
  4.     .stream() 
  5.     .map(request-> 
  6.         CompletableFuture.supplyAsync(e->{ 
  7.             //some opts 
  8.         },executor)) 
  9.     .collect(Collectors.toList()); 
  10.  
  11. CompletableFuture allCF = CompletableFuture.allOf(futureList.toArray(new CompletableFuture[0])); 
  12.  
  13. allCF.join(); 

我们这里用到了一个主要的函数,那就是allOf,用来把所有的CompletableFuture组合在一起;类似的还有anyOf,表示只运行其中一个。常用的,还有三个函数:

  • thenAcceptBoth 处理两个任务的情况,有两个任务结果入参,无返回值
  • thenCombine 处理两个任务的情况,有入参有返回值,最喜欢
  • runAfterBoth 处理两个任务的情况,无入参,无返回值

End

自从认识了CompletableFuture,我已经很少硬编码Future了。相对于各种回调的嵌套,CompletableFuture为我们提供了更直观、更优美的API。在“多个任务等待完成状态”这个应用场景,CompletableFuture已经成了我的首选。

唯一的问题是,它的函数有点多,你需要熟悉一小段时间。另外,有一个小小的问题,个人觉得,这个类如果叫做Promise的话,就能够和JS的统一起来,算是锦上添花吧。

 

作者简介:小姐姐味道 (xjjdog),一个不允许程序员走弯路的公众号。聚焦基础架构和Linux。十年架构,日百亿流量,与你探讨高并发世界,给你不一样的味道。

 


文章名称:CompletableFuture真香,可以替代CountDownLatch!
路径分享:http://cdxtjz.cn/article/cojsdcd.html

联系我们

您好HELLO!
感谢您来到成都网站建设公司,若您有合作意向,请您为我们留言或使用以下方式联系我们, 我们将尽快给你回复,并为您提供真诚的设计服务,谢谢。
  • 电话:028- 86922220 18980695689
  • 商务合作邮箱:631063699@qq.com
  • 合作QQ: 532337155
  • 成都网站设计地址:成都市青羊区锣锅巷31号五金站写字楼6楼

小谭建站工作室

成都小谭网站建设公司拥有多年以上互联网从业经验的团队,始终保持务实的风格,以"帮助客户成功"为已任,专注于提供对客户有价值的服务。 我们已为众企业及上市公司提供专业的网站建设服务。我们不只是一家网站建设的网络公司;我们对营销、技术、管理都有自己独特见解,小谭建站采取“创意+综合+营销”一体化的方式为您提供更专业的服务!

小谭观点

相对传统的成都网站建设公司而言,小谭是互联网中的网站品牌策划,我们精于企业品牌与互联网相结合的整体战略服务。
我们始终认为,网站必须注入企业基因,真正使网站成为企业vi的一部分,让整个网站品牌策划体系变的深入而持久。