手把手教学妹CompletableFuture异步化,性能关系直接起飞!(下)

简介: 手把手教学妹CompletableFuture异步化,性能关系直接起飞!

thenAccept系

image.png

参数consumer的类型是接口Consumer<T>,这个接口里与CompletionStage相关的方法是

void accept(T t)

image.png该方法虽然支持参数,但不支持返回值,所以thenAccept系方法返回值是CompletionStage。

thenRun系

image.png

参数是Runnable,所以action既不能接收参数也不支持返回值,所以thenRun系列方法返回的也是CompletionStage<Void>

Async表示异步执行fn、consumer或action。

thenCompose系

这个系列的方法会新创建出一个子流程,最终结果和thenApply系相同。

看如何使用thenApply()。

supplyAsync()启动一个异步流程,之后是两个串行操作。虽然这是一个异步流程,但任务1、2、3是串行执行,即2依赖1的执行结果,3依赖2的执行结果。

image.png

2 AND汇聚

主要是thenCombine、thenAcceptBoth和runAfterBoth系接口

3 OR汇聚

主要是applyToEither、acceptEither和runAfterEither系接口

CompletionStage applyToEither(other, fn);
CompletionStage applyToEitherAsync(other, fn);
CompletionStage acceptEither(other, consumer);
CompletionStage acceptEitherAsync(other, consumer);
CompletionStage runAfterEither(other, action);
CompletionStage runAfterEitherAsync(other, action);

如何使用applyToEither()描述OR汇聚关系。

CompletableFuture<String> f1 = 
  CompletableFuture.supplyAsync(()->{
    int t = getRandom(5, 10);
    sleep(t, TimeUnit.SECONDS);
    return String.valueOf(t);
});
CompletableFuture<String> f2 = 
  CompletableFuture.supplyAsync(()->{
    int t = getRandom(5, 10);
    sleep(t, TimeUnit.SECONDS);
    return String.valueOf(t);
});
CompletableFuture<String> f3 = 
  f1.applyToEither(f2,s -> s);
System.out.println(f3.join());

CompletableFuture 中各种关系(并行、串行、聚合),支持的各种场景。 比如:线程A 等待线程B或线程C等待线程A、B 。


其实CountdownLatch、ThreadPoolExecutor 和Future 就是来解决这些关系场景的,现在有了 completableFuture,可以优先考虑使用 CompletableFuture。

4 异常处理

fn、consumer、action的核心方法都不允许抛受检异常,但无法限制它们抛运行时异常,例如下面的代码,执行 1/0 就会出现除0错误的运行时异常

非异步编程里,可以用try/catch捕获并处理异常,异步编程里该如何处理呢?

image.png

CompletionStage给出的方案很简单,使用这些方法处理异常和串行操作一样的,而且还支持链式编程。

CompletionStage exceptionally(fn);
CompletionStage<R> whenComplete(consumer);
CompletionStage<R> whenCompleteAsync(consumer);
CompletionStage<R> handle(fn);
CompletionStage<R> handleAsync(fn);
  • exceptionally()类似try/catch中的catch

image.png

  • whenComplete()和handle()类似try/finally的finally,无论是否发生异常都会执行whenComplete()中的回调方法consumer和handle()中的回调方法fn
    whenComplete()不支持返回结果,handle()支持返回结果。


学了这么多,最后来看个例子:

//采购订单
PurchersOrder po;
CompletableFuture<Boolean> cf = 
  CompletableFuture.supplyAsync(()->{
    // 在MySQL中查询规则
    return findRuleByJdbc();
  }).thenApply(r -> {
    // 规则校验
    return check(po, r);
});
Boolean isOk = cf.join();

如上代码问题在于:

  • 读数据库属于I/O操作,应定制单独的线程池,避免线程饥饿
  • 查出来的结果做为下一步处理的条件,若结果为空,没有对应处理
  • 异常未处理
目录
相关文章
|
5月前
|
Java
领略Lock接口的风采,通过实战演练,让你迅速掌握这门高深武艺,成为Java多线程领域的武林盟主
领略Lock接口的风采,通过实战演练,让你迅速掌握这门高深武艺,成为Java多线程领域的武林盟主
57 7
|
8月前
|
Java
JAVA多线程深度解析:线程的创建之路,你准备好了吗?
【6月更文挑战第19天】Java多线程编程提升效率,通过继承Thread或实现Runnable接口创建线程。Thread类直接继承启动简单,但限制多继承;Runnable接口实现更灵活,允许类继承其他类。示例代码展示了两种创建线程的方法。面对挑战,掌握多线程,让程序高效运行。
36 1
|
9月前
|
缓存 算法 编译器
【C/C++ 泡沫精选面试题01】提高c++性能,你用过哪些方式去提升?
【C/C++ 泡沫精选面试题01】提高c++性能,你用过哪些方式去提升?
141 1
|
9月前
|
消息中间件 存储 算法
【C/C++ 泡沫精选面试题04】在实际项目中,多进程和多线程如何选择?
【C/C++ 泡沫精选面试题04】在实际项目中,多进程和多线程如何选择?
265 1
|
9月前
|
缓存 算法 Java
堪称神级的阿里巴巴“高并发”教程《基础+实战+源码+面试+架构》
作为一个普普通通的程序员,如何才能提升自己的能力,在职场上拥有一技之长,这也成为普通的你我,迫切的需求。
|
Java 程序员
终于不慌内卷了,多亏阿里内部的并发图册+JDK源码速成笔记
并发编程 Java并发在近几年的面试里面可以说是面试热点,每个面试官面试的时候都会跟你扯一下并发,甚至是高并发。面试前你不仅得需要弄清楚的是什么是并发,还得搞清什么是高并发! 在这里很多小白朋友就会很疑惑:我工作又不用,为啥面试总是问?真就内卷卷我呗!(手动狗头)互联网内卷已经是现在的行业趋势,而且是不可逆的,这个大家也知道;但LZ要说的是,虽然简单地增删改查并不需要并发的知识,但是业务稍微复杂一点,你的技术水平稍微提升一点的话你就会知道,并发是我们Java程序员绕不开的一道坎。
68 0
|
设计模式 缓存 负载均衡
你kin你擦!阿里终于肯把内部高并发编程高阶笔记开源出来了
“高并发”三字是近几年开发圈子里热议的一个话题,可能程序员之间闲下来就会讨论所谓的“高并发经验”。值得注意的是即使你和高并发天天打交道,也不一定能获得高并发的经验,高并发只是一个结果,并不是过程。想要玩转高并发,基础最重要,大并发面前,靠得住的只有人,是人来根据具体的应用场景去解决具体的问题。
你kin你擦!阿里终于肯把内部高并发编程高阶笔记开源出来了
直击灵魂!美团大牛手撸并发原理笔记,由浅入深剖析JDK源码
并发编程这四个字想必大家最近都在网上看到过有很多的帖子在讨论。我们都知道并发编程可选择的方式有多进程、多线程和多协程。在Java中,并发就是多线程模式。而多线程编程也一直是一个被广泛而深入讨论的领域。如果遇到复杂的多线程编程场景,大多数情况下我们就需要站在巨人的肩膀上利用并发编程框架——JDK Concurrent包来解决相关线程问题。
面试官:小伙子我们先来唠唠并发编程的几大核心知识点
并发编程算是Java的一个难点,经常做业务相关的程序员基本上用不到juc的包,但是这些知识点十分重要,所以不管在哪里,时刻保持学习真的很重要。
|
消息中间件 JavaScript 小程序
一网打尽:异步神器 CompletableFuture 万字详解!
一网打尽:异步神器 CompletableFuture 万字详解!