一、背景

jdk8中加入了实现类CompletableFuture,用于异步编程。底层做任务使用的是ForkJoin, 顾名思义,是将任务的数据集分为多个子数据集,而每个子集,都可以由独立的子任务来处理,最后将每个子任务的结果汇集起来。它是ExecutorService接口的一个实现,它把子任务分配给线程池(称为ForkJoinPool)中的工作线程。从api文档看,它实现了2个接口CompletionStage和Future。CompletionStage支持lambda表达式,接口的方法的功能都是在某个阶段得到结果后要做的事情。因此,CompletableFuture不仅拥有Future的所有特性,而且还内置了lambda表达式,支持异步回调,结果转换等功能,它有以下Future实现不了的功能:

  1. 合并两个相互独立的异步计算的结果

  2. 等待异步任务的所有任务都完成

  3. 等待异步任务的其中一个任务完成就返回结果

  4. 任务完成后调用回调方法

  5. 任务完成的结果可以用于下一个任务。

  6. 任务完成时发出通知提供原生的异常处理api

二、代码

package com.example.demo;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.*; public class CompletableFutureDemo {
//CPU核数
private static final int AVAILABLE_PROCESSORS = Runtime.getRuntime().availableProcessors();
private static final ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(AVAILABLE_PROCESSORS,
3 * AVAILABLE_PROCESSORS,
3, TimeUnit.SECONDS,
new LinkedBlockingDeque<>(20)); public static void main(String[] args) throws Exception {
long startTime = System.currentTimeMillis();
System.out.println("demo start....." + startTime);
demo3();
System.out.println("demo end.....costTime = " + (System.currentTimeMillis() - startTime));
} /**
* 基于allOf,并行处理多个任务,等待所有任务执行完毕后返回
*/ public static void demo3() throws Exception {
//用户整体接收各个任务的返回值
Map<String,String> dataMap = new ConcurrentHashMap<>();
List<CompletableFuture<String>> futureList = new ArrayList<>();
futureList.add(doSomethingA("A", dataMap));
futureList.add(doSomethingB("B", dataMap));
futureList.add(doSomethingC("C", dataMap));
CompletableFuture<Void> result = CompletableFuture.allOf(futureList.toArray(new CompletableFuture[0]));
try {
result.get(3, TimeUnit.SECONDS);
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("dataMap = " + dataMap);
//结果为:{doSomeThingB=B, doSomeThingA=A}
} /**
* 基于thenCompose,第一个任务执行完后,第二个任务使用第一个任务的返回作为参数
*/
public static void demo1() throws Exception {
Map<String,String> dataMap = new HashMap<>();
CompletableFuture<String> completableFuture = doSomethingA("A", dataMap)
.thenCompose(id -> doSomethingB(id, dataMap));
String result = completableFuture.get(3, TimeUnit.SECONDS);
System.out.println("result = " + result);
//结果为:A is done is done } /**
* 基于thenCombine,当两个任务都完成后,使用两者的结果作为参数再执行一个异步任务
*/
public static void demo2() throws Exception {
Map<String,String> dataMap = new HashMap<>();
CompletableFuture<String> completableFuture = doSomethingA("A", dataMap)
.thenCombine(doSomethingB("B", dataMap), (a, b) -> a + " - " + b);
String result = completableFuture.get(3, TimeUnit.SECONDS);
System.out.println("result = " + result);
//结果为:A is done - B is done
} /**
* @param dataMap 用户整体接收方法的返回值
* @return
*/
public static CompletableFuture<String> doSomethingA(String taskId, Map<String,String> dataMap) {
System.out.println("doSomethingA start....." + System.currentTimeMillis());
return CompletableFuture.supplyAsync(() -> {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
dataMap.put("doSomeThingA", "A");
System.out.println(taskId + " is done and dataMap"+dataMap);
return taskId + " is done";
}, threadPoolExecutor);
} public static CompletableFuture<String> doSomethingB(String taskId, Map<String,String> dataMap) {
System.out.println("doSomethingB start....." + System.currentTimeMillis());
return CompletableFuture.supplyAsync(() -> {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
dataMap.put("doSomeThingB", "B");
System.out.println(taskId + " is done and dataMap"+dataMap);
return taskId + " -> B is done";
}, threadPoolExecutor);
} public static CompletableFuture<String> doSomethingC(String taskId, Map<String,String> dataMap) {
System.out.println("doSomethingC start....." + System.currentTimeMillis());
return CompletableFuture.supplyAsync(() -> {
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
dataMap.put("doSomeThingC", "C");
System.out.println(taskId + " is done and dataMap"+dataMap);
return taskId + " is done";
}, threadPoolExecutor); } }

三、效率比较

很明显,异步更快

package com.example.demo;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.CompletableFuture; /**
* @author d00018641
* @date 2021/11/4 15:10
*/
public class TestDemo2 {
private static final String key = "llllllllllllllllllllllll";
public static void main(String[] args) { List<String> requestList = new ArrayList<>();
requestList.add("3");
requestList.add("4");
requestList.add("5");
requestList.add("6");
// 响应参数list
String[] returnArray = new String[requestList.size()];
// 异步查询每一列,定义响应列数的futures
List<CompletableFuture<String>> futures = new ArrayList<>();
long startTime = System.currentTimeMillis();
for (int i = 0; i < requestList.size(); i++) {
final int a = i;
CompletableFuture<String> tf = CompletableFuture.supplyAsync(() -> {
return calc(requestList.get(a));
}).whenComplete((m, e) -> returnArray[a] = m);
futures.add(tf);
}
CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).join();
//CompletableFuture end.....costTime = 147
System.out.println("CompletableFuture end.....costTime = " + (System.currentTimeMillis() - startTime));
long startTime1 = System.currentTimeMillis();
for(int i = 0; i < requestList.size(); i++){
returnArray[i] = calc(requestList.get(i));
}
//连续 end.....costTime = 432
System.out.println("连续 end.....costTime = " + (System.currentTimeMillis() - startTime1));
System.out.println(Arrays.asList(returnArray)); } private static String calc(String source) {
int as = Integer.parseInt(source);
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
return String.valueOf(Math.pow(as, 3));
}
}

四、编程实战

Map<String, ExchangeRateVO> cacheMap = new ConcurrentHashMap<>();
List<CompletableFuture<Void>> batchFutureList = new ArrayList<>();
for (ExchangeRateVO vo : paramList) {
CompletableFuture<Void> batchItem = CompletableFuture.runAsync(() -> {
List<CompletableFuture<Void>> itemFutureList = new ArrayList<>();
for (LookupItemVO toCurrency : toCurrencyList) {
CompletableFuture<Void> cfItem = CompletableFuture.runAsync(() -> {
// 查询接口返回汇率数据
ExchangeRateVO resultVo;
try {
buildBasicContext();
resultVo = iExchangeRateService.findExchangeRate(vo.getCountryCode(), vo.getFromCurrency(),
toCurrency.getItemCode(), vo.getStartDate());
if (CommonUtils.isNotEmpty(resultVo)) {
if (CommonUtils.isEmpty(resultVo.getFromCurrency())) {
resultVo.setFromCurrency(vo.getFromCurrency());
resultVo.setToCurrency(toCurrency.getItemCode());
resultVo.setCountryCode(vo.getCountryCode());
}
resultVo.setStartDate(vo.getStartDate());
resultList.add(resultVo);
cacheMap.put(vo.getCountryCode() + vo.getFromCurrency() + toCurrency.getItemCode(),
resultVo);
}
} catch (Exception er) {
LOGGER.error("findExchangeRate has some error:", er);
}
});
itemFutureList.add(cfItem);
}
CompletableFuture.allOf(itemFutureList.toArray(new CompletableFuture[itemFutureList.size()])).join();
});
batchFutureList.add(batchItem);
}
LOGGER.info2("before submit completable");
CompletableFuture.allOf(batchFutureList.toArray(new CompletableFuture[batchFutureList.size()])).join();

JDK8的异步处理方式-CompletableFuture的使用的更多相关文章

  1. Java8 异步编排类CompletableFuture

    为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. https://www.cnblogs.com/shijiaqi1066/p/8758206 ...

  2. 异步技巧之CompletableFuture

    摘自--https://juejin.im/post/5b4622df5188251ac9766f47 异步技巧之CompletableFuture 1.Future接口 1.1 什么是Future? ...

  3. Java 异步编程 (5 种异步实现方式详解)

    ​ 同步操作如果遇到一个耗时的方法,需要阻塞等待,那么我们有没有办法解决呢?让它异步执行,下面我会详解异步及实现@mikechen 目录 什么是异步? 一.线程异步 二.Future异步 三.Comp ...

  4. js中同步与异步请求方式

    异步请求方式: $.ajax({ url : 'your url', data:{name:value}, cache : false, async : true, type : "POST ...

  5. iOS开发——网络编程Swift篇&(六)异步Post方式

    异步Post方式 // MARK: - 异步Post方式 func asynchronousPost() { //创建NSURL对象 var url:NSURL! = NSURL(string: &q ...

  6. iOS开发——网络编程Swift篇&(四)异步Get方式

    异步Get方式 // MARK: - 异步Get方式 func asynchronousGet() { //创建NSURL对象 var url:NSURL! = NSURL(string: " ...

  7. Delphi中ADO异步执行方式

    当ADO开始处理数据后,应用程序必须等到ADO处理完毕之后才可以继续执行.但是除了同步执行方式之外,ADO也提供了异步执行的方式,允许当ADO处理时,应用程序仍然能够先继续执行.而当ADO处理数据完毕 ...

  8. 前端Mahsup异步依赖方式不能做业务数据依赖

    很久之前流行mashup方式做内容集成,之前为了IP定位的方便,引用了第三方的IP定位JS,然后根据其内容与服务器同步地址数据并写入Cookie,可是这种方式一旦,第三方的库反应缓慢时,就会出现大问题 ...

  9. 《C# 爬虫 破境之道》:第一境 爬虫原理 — 第四节:同步与异步请求方式

    前两节,我们对WebRequest和WebResponse这两个类做了介绍,但两者还相对独立.本节,我们来说说如何将两者结合起来,方式有哪些,有什么不同. 1.4.1 说结合,无非就是我们如何发送一个 ...

  10. 【JDK8】Java8 优雅的异步调用API CompletableFuture

    1.CompletableFuture是什么? CompletableFuture是JDK8的新特性之一,是异步调用相关的API,用于简化异步调用,提高异步调用的效率 2.CompletableFut ...

随机推荐

  1. filebeat+Elk实现日志收集并使用kibana展示

    工作流程图 通过Filegeat收集日志,将日志的数据推送到kafka然后通过logstash去消费发送到Es,再通过索引的方式将数据用kibana进行展示: 1.部署测试机器规划 ip       ...

  2. Linux 中设置静态的IP

    Topic (What i going to write) Setting the static Ip address in linux Main Information Firstly use CD ...

  3. 在Excel中创建随机数据集

    1.随机小数0-1之间 =RAND() 2.随机整数1-100之间 =RANDBETWEEN(1,100) 3.生成一定比例的随机数0或1 =IF(RAND()>=0.8,1,0) 4.生成一定 ...

  4. [Swift]创建桥接文件,Swift使用MJRefresh刷新插件

    刚开始玩Swift,想做个下拉刷新的功能,发现在OC中用得比较多的第三方插件是MJRefresh.查了一下,在Swift中使用OC的插件要通过桥接文件,然后又百度一下怎么创建桥接文件,发现很多都是老司 ...

  5. Windows 分辨率改变引起WPF的适应问题

    系统分辨率变化,全屏WPF的UI出现没更新和白底之类的现象记录. 您可以使用SystemParameters.WorkArea.最初设置MainWindow的MaxHeight. 在MainWindo ...

  6. Java-如何打包下载成.zip文件

    打包下载成.zip文件 项目背景 公司使用vue + SpringBoot实现批量下载功能 今天在调试批量下载这个功能.打包成.zip文件时,在返回给前端浏览器出现报错信息: 后端报错: ERROR ...

  7. java轻量级锁、重量级锁、可重入锁、偏向锁、自旋锁的概念

    1.重量级锁 作为互斥同步的方式,是最基础的锁,其他的锁都是为了减少开销做的优化,重量级锁借助了monitor 对象,monitor对象中有三个区域,分别是entity site. owner和wai ...

  8. python桌面应用自动化,uiautomation模块的Depth和searchDepth心得

    最近在学习yinkaisheng大神写的uiautomation模块,Depth和searchDepth一直使用不好,明明Depth=3,居然可以用searchDepth=1找到,网上也没找到答案,就 ...

  9. 文本超出换行添加white-space:wrap无效

    场景描述: 在vue项目中,在Modal弹窗里面使用Form表单组件,然后在FormItem里面放一个div标签用来装文字内容.有时会出现内容超出Form表单宽度但是不换行的问题. 解决方法: 给di ...

  10. 火星坐标系 (GCJ-02) 与百度坐标系 (BD-09) 的互转

    /**火星坐标系 (GCJ-02) 与百度坐标系 (BD-09) 的互转  * Created by macremote on 16/5/3.  */ function GPSUtil (){   v ...