聊聊JDK8的CompletableFuture,你明白了吗?

CompletableFuture

CompletableFuture是java.util.concurrent库在java 8中新增的主要工具,同传统的Future相比,其支持流式计算、函数式编程、完成通知、自定义异常处理等很多新的特性。

supplyAsync方法

通过该函数创建的CompletableFuture实例会异步执行当前传入的计算任务。在调用端,则可以通过get或join获取最终计算结果。

这个有两个不同的实现方式,一种是我们传入我们自己创建的线程池,然后使用我们创建的线程池进行操作,还有一种就是不传线程池,让程序是使用默认的线程池进行操作。

//使用默认线程池
public static CompletableFuture supplyAsync(Supplier supplier) {
return asyncSupplyStage(asyncPool, supplier);
}
//使用自定义线程池
public static CompletableFuture supplyAsync(Supplier supplier, Executor executor) {
return asyncSupplyStage(screenExecutor(executor), supplier);
}

第一种只需传入一个Supplier实例(一般使用lamda表达式),此时框架会默认使用ForkJoin的线程池来执行被提交的任务。

我们来自定义一个代码看一下:

ExecutorService executors = Executors.newFixedThreadPool(5);

CompletableFuture> bFuture = CompletableFuture.supplyAsync(() -> {
//执行查询
QueryWrapper queryWrapper = new QueryWrapper<>();
queryWrapper.eq("userId",params.getUserId());
queryWrapper.eq("SHOP_ID",params.getShopId());
List list = orderService.list(queryWrapper);
return list;
}, executors);

当我们执行查询的时候,这时候实际上就属于异步的查询的,我们可以写多个查询,比如,上面的代码我们查询的是订单,下面我们可以查询用户的信息,还是使用同样的线程池。

       CompletableFuture> aFuture = CompletableFuture.supplyAsync(() -> {
//执行查询
QueryWrapper queryWrapper = new QueryWrapper<>();
queryWrapper.eq("SHOP_ID",params.getShopId());
List list = userService.list(queryWrapper);
return list;
}, executors);

这时候,我们就可以获取一下这个返回的结果,那么获取返回结果我们需要调用什么方法呢?

可以通过get或join获取最终计算结果。这时候我们可以打印一下,获取结果的长度。

List list = aFuture.get();
List list = bFuture.get();

这样,我们就可以使用异步查询来完成我们对结果集的查询。

这就有小伙伴想问,如果我想要用第一个的结果,去在第二个数据中去查询,我们应该怎么做呢?

thenApply

thenApply提交的任务类型需遵从Function签名,也就是有入参和返回值,其中入参为前置任务的结果。

什么意思呢?其实很简单,我们这里直接看代码:

    CompletableFuture> cFuture = bFuture.thenApply((list) -> {
List collect = list.stream().map(User::getUserId).collect(Collectors.toList());
...
return orderList;
});

这实际上,就是我们根据查询出的所有用户的集合,直接获取到他的userId,然后我们根据UserId,把这些用户下的订单数据都提取出来,当然,在实际使用中,我们理论上可以无限连接后续计算任务,从而实现链条更长的流式计算。

但是这种链式也不是都非常的好用,毕竟要控制住线程池,大家记得在使用完成之后,可以把自己创建的线程池小回调,调用shutDown方法就可以了。我们再接着往下说。

thenAccept

实际上thenAccept的效果,和thenApply 的效果等同,但是thenAccept提交的任务类型需遵从Consumer签名,也就是有入参但是没有返回值,其中入参为前置任务的结果。

实际上调用的是和之前一样的,但是就是没有返回值了。

 CompletableFuture cFuture = bFuture.thenAccept((list) -> {
List collect = list.stream().map(User::getUserId).collect(Collectors.toList());
...

});

这就是区别,不需要进行返回了,这种一般用的地方比较特殊,就比如说我们执行了很多操作,但是在其中需要夹杂一些比如说给某个服务进行通知,但是我们并不需要这个服务给我们返回值的结果的时候,实际上完全可以使用这种方式。方便还快捷。当然,你想直接调用,阿粉也是没有意见的。

whenComplete

whenComplete主要用于注入任务完成时的回调通知逻辑。这个解决了传统future在任务完成时,无法主动发起通知的问题。前置任务会将计算结果或者抛出的异常作为入参传递给回调通知函数。

这个方法就是相当于是把前一个任务中的结果,通过第二个方法获取结果,也并不会影响第二个的逻辑。

示例代码如下:

CompletableFuture> listCompletableFuture = bFuture.whenComplete((r, e) -> {
if (e != null) {
System.out.println("compute failed!");
} else {
System.out.println("received result is " + r);
}
});

实际上最后答应的就是接受者的信息。大家学会了么?

网页标题:聊聊JDK8的CompletableFuture,你明白了吗?
URL网址:http://www.mswzjz.cn/qtweb/news41/378741.html

攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能