JDK8的异步处理方式-CompletableFuture的使用
一、背景
jdk8中加入了实现类CompletableFuture,用于异步编程。底层做任务使用的是ForkJoin, 顾名思义,是将任务的数据集分为多个子数据集,而每个子集,都可以由独立的子任务来处理,最后将每个子任务的结果汇集起来。它是ExecutorService接口的一个实现,它把子任务分配给线程池(称为ForkJoinPool)中的工作线程。从api文档看,它实现了2个接口CompletionStage和Future。CompletionStage支持lambda表达式,接口的方法的功能都是在某个阶段得到结果后要做的事情。因此,CompletableFuture不仅拥有Future的所有特性,而且还内置了lambda表达式,支持异步回调,结果转换等功能,它有以下Future实现不了的功能:
合并两个相互独立的异步计算的结果
等待异步任务的所有任务都完成
等待异步任务的其中一个任务完成就返回结果
任务完成后调用回调方法
任务完成的结果可以用于下一个任务。
任务完成时发出通知提供原生的异常处理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的使用的更多相关文章
- Java8 异步编排类CompletableFuture
为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/ShiJiaqi. https://www.cnblogs.com/shijiaqi1066/p/8758206 ...
- 异步技巧之CompletableFuture
摘自--https://juejin.im/post/5b4622df5188251ac9766f47 异步技巧之CompletableFuture 1.Future接口 1.1 什么是Future? ...
- Java 异步编程 (5 种异步实现方式详解)
同步操作如果遇到一个耗时的方法,需要阻塞等待,那么我们有没有办法解决呢?让它异步执行,下面我会详解异步及实现@mikechen 目录 什么是异步? 一.线程异步 二.Future异步 三.Comp ...
- js中同步与异步请求方式
异步请求方式: $.ajax({ url : 'your url', data:{name:value}, cache : false, async : true, type : "POST ...
- iOS开发——网络编程Swift篇&(六)异步Post方式
异步Post方式 // MARK: - 异步Post方式 func asynchronousPost() { //创建NSURL对象 var url:NSURL! = NSURL(string: &q ...
- iOS开发——网络编程Swift篇&(四)异步Get方式
异步Get方式 // MARK: - 异步Get方式 func asynchronousGet() { //创建NSURL对象 var url:NSURL! = NSURL(string: " ...
- Delphi中ADO异步执行方式
当ADO开始处理数据后,应用程序必须等到ADO处理完毕之后才可以继续执行.但是除了同步执行方式之外,ADO也提供了异步执行的方式,允许当ADO处理时,应用程序仍然能够先继续执行.而当ADO处理数据完毕 ...
- 前端Mahsup异步依赖方式不能做业务数据依赖
很久之前流行mashup方式做内容集成,之前为了IP定位的方便,引用了第三方的IP定位JS,然后根据其内容与服务器同步地址数据并写入Cookie,可是这种方式一旦,第三方的库反应缓慢时,就会出现大问题 ...
- 《C# 爬虫 破境之道》:第一境 爬虫原理 — 第四节:同步与异步请求方式
前两节,我们对WebRequest和WebResponse这两个类做了介绍,但两者还相对独立.本节,我们来说说如何将两者结合起来,方式有哪些,有什么不同. 1.4.1 说结合,无非就是我们如何发送一个 ...
- 【JDK8】Java8 优雅的异步调用API CompletableFuture
1.CompletableFuture是什么? CompletableFuture是JDK8的新特性之一,是异步调用相关的API,用于简化异步调用,提高异步调用的效率 2.CompletableFut ...
随机推荐
- react修改打包后css,js,src引入的静态地址路径修改
在package.json 里边增加该配置 (当然,配置的地址可以由部署环境决定,可以是 './' , 也可以是 '../',根据需要配置即可) 查找到到"private": tr ...
- 错误提示“com.alibaba.fastjson.JSONException: exepct '[', but string, pos 4, json”解决
1.错误提示信息如下: com.alibaba.fastjson.JSONException: exepct '[', but string, pos 4, json : "[{" ...
- Linuxt通过命令lsof或者extundelete工具恢复误删除的文件或者目录
Linux不像windows有那么显眼的回收站,不是简单的还原就可以了.linux删除文件还原可以分为两种情况,一种是删除以后在进程存在删除信息,一种是删除以后进程都找不到,只有借助于工具还原.这里分 ...
- wireshark作业
1Wireshark基本操作: 1.启动wireshark,正确选择混杂模式,访问任意网站: 2.设置过滤器呈现本地和该网站服务器之间的交互报文: 3.保存抓包结果文件.cap: 4.在作业纸上记录下 ...
- VSCode 设置启用终端执行yarn
如果你的vscode终端不支持 yarn run * 的命令你可以新建终端执行:get-ExecutionPolicy执行结果为:RemoteSigned 表示为禁用状态. 执行:set-Execut ...
- QT使用中出现的问题
1.运行程序程序弹出The CDB process terminated 2.调试弹出窗口提示缺少qtcreatorcdbext.dll 1.运行程序程序弹出The CDB process termi ...
- java-javaSE-集合类
集合类结构 集合类的基本接口是 Collection 和 Map Collection 向下派生出 Set List Queue 等接口 Map 向下派生出 HashMap LinkedHashMap ...
- [Docker-1自顶向下学习Docker
本文目录: 什么是DOCKER? 什么是容器? 什么是DOCKER镜像? DOCKER有什么使用场景和优势? 流程图一:从中央仓库拉取镜像并部署 流程图二:上传镜像到中央私库 结语 什么是DOCK ...
- PY3多继承
__author__ = "Alex Li"class A: def __init__(self): print("A")class B(A): pass #d ...
- 入坑winpdb-1.4.8
这几天莫名的不能在docker中使用winpdb-1.4.8调试,把容器重新安装了也不行,跟踪调试以后发现原因是容器中使用pycryptodemo作为rpdb2的加密库,加密向量IV需要以bytes类 ...