官方文档:
https://gitee.com/jd-platform-opensource/asyncTool/blob/master/QuickStart.md
场景
之前做项目的时候面临一个场景:一个业务操作有很多异步任务,任务与任务之间又有关联,如下图的例子:
多个执行单元的串行请求

多个执行单元的并行请求

阻塞等待,串行的后面跟多个并行

阻塞等待,多个并行的执行完毕后才执行某个

串并行相互依赖

更复杂的场景

对于上面的场景我尝试了多种方式:
- JDK8的CompleteFuture
2. Reactive
3. Future 
传统的Future、CompleteableFuture一定程度上可以完成任务编排,并可以把结果传递到下一个任务。如CompletableFuture有then方法,但是却无法做到对每一个执行单元的回调。譬如A执行完毕成功了,后面是B,我希望A在执行完后就有个回调结果,方便我监控当前的执行状况,或者打个日志什么的。失败了,我也可以记录个异常信息什么的。
此时,CompleteableFuture就无能为力了。
AsyncTool框架提供了这样的回调功能。并且,如果执行异常、超时,可以在定义这个执行单元时就设定默认值
我用了asyncTool,任务与任务的关系变简单了,我能随意编排任务。
实战演示
A执行完毕后,开启另外两个B和C,另外两个执行完毕后,开始执行F
引入maven依赖
<dependency><groupId>com.gitee.jd-platform-opensource</groupId><artifactId>asyncTool</artifactId><version>V1.4-SNAPSHOT</version></dependency>
基本组件
worker: 一个最小的任务执行单元。通常是一个网络调用,或一段耗时操作。
T,V两个泛型,分别是入参和出参类型。
譬如该耗时操作,入参是String,执行完毕的结果是Integer,那么就可以用泛型来定义。
多个不同的worker之间,没有关联,分别可以有不同的入参、出参类型。
/*** 每个最小执行单元需要实现该接口* @author wuweifeng wrote on 2019-11-19.*/public interface IWorker<T, V> {/*** 在这里做耗时操作,如rpc请求、IO等** @param object* object*/V action(T object, Map<String, WorkerWrapper> allWrappers);/*** 超时、异常时,返回的默认值* @return 默认值*/V defaultValue();}
callBack:对每个worker的回调。worker执行完毕后,会回调该接口,带着执行成功、失败、原始入参、和详细的结果。
/*** 每个执行单元执行完毕后,会回调该接口</p>* 需要监听执行结果的,实现该接口即可* @author wuweifeng wrote on 2019-11-19.*/public interface ICallback<T, V> {void begin();/*** 耗时操作执行完毕后,就给value注入值**/void result(boolean success, T param, WorkResult<V> workResult);}
wrapper:组合了worker和callback,是一个 最小的调度单元 。通过编排wrapper之间的关系,达到组合各个worker顺序的目的。
wrapper的泛型和worker的一样,决定了入参和结果的类型。
WorkerWrapper<String, String> workerWrapper = new WorkerWrapper<>(w, "0", w);WorkerWrapper<String, String> workerWrapper1 = new WorkerWrapper<>(w1, "1", w1);WorkerWrapper<String, String> workerWrapper2 = new WorkerWrapper<>(w2, "2", w2);WorkerWrapper<String, String> workerWrapper3 = new WorkerWrapper<>(w3, "3", w3);
如
0执行完,同时1和2, 1\2都完成后3。3会等待2完成
此时,你可以定义一个 worker
/*** @author wuweifeng wrote on 2019-11-20.*/public class ParWorker1 implements IWorker<String, String>, ICallback<String, String> {@Overridepublic String action(String object) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}return "result = " + SystemClock.now() + "---param = " + object + " from 1";}@Overridepublic String defaultValue() {return "worker1--default";}@Overridepublic void begin() {//System.out.println(Thread.currentThread().getName() + "- start --" + System.currentTimeMillis());}@Overridepublic void result(boolean success, String param, WorkResult<String> workResult) {if (success) {System.out.println("callback worker1 success--" + SystemClock.now() + "----" + workResult.getResult()+ "-threadName:" +Thread.currentThread().getName());} else {System.err.println("callback worker1 failure--" + SystemClock.now() + "----" + workResult.getResult()+ "-threadName:" +Thread.currentThread().getName());}}}
通过这一个类看一下,action里就是你的耗时操作,begin就是任务开始执行时的回调,result就是worker执行完毕后的回调。当你组合了多个执行单元时,每一步的执行,都在掌控之内。失败了,还会有自定义的默认值。这是CompleteableFuture无法做到的。
安装教程
使用说明
一.3个任务并行

ParWorker w = new ParWorker();ParWorker1 w1 = new ParWorker1();ParWorker2 w2 = new ParWorker2();WorkerWrapper<String, String> workerWrapper2 = new WorkerWrapper.Builder<String, String>().worker(w2).callback(w2).param("2").build();WorkerWrapper<String, String> workerWrapper1 = new WorkerWrapper.Builder<String, String>().worker(w1).callback(w1).param("1").build();WorkerWrapper<String, String> workerWrapper = new WorkerWrapper.Builder<String, String>().worker(w).callback(w).param("0").build();long now = SystemClock.now();System.out.println("begin-" + now);Async.beginWork(1500, workerWrapper, workerWrapper1, workerWrapper2);// Async.beginWork(800, workerWrapper, workerWrapper1, workerWrapper2);// Async.beginWork(1000, workerWrapper, workerWrapper1, workerWrapper2);System.out.println("end-" + SystemClock.now());System.err.println("cost-" + (SystemClock.now() - now));System.out.println(Async.getThreadCount());System.out.println(workerWrapper.getWorkResult());Async.shutDown();
二.1个执行完毕后,开启另外两个,另外两个执行完毕后,开始第4个

ParWorker w = new ParWorker();ParWorker1 w1 = new ParWorker1();ParWorker2 w2 = new ParWorker2();ParWorker3 w3 = new ParWorker3();WorkerWrapper<String, String> workerWrapper3 = new WorkerWrapper.Builder<String, String>().worker(w3).callback(w3).param("3").build();WorkerWrapper<String, String> workerWrapper2 = new WorkerWrapper.Builder<String, String>().worker(w2).callback(w2).param("2").next(workerWrapper3).build();WorkerWrapper<String, String> workerWrapper1 = new WorkerWrapper.Builder<String, String>().worker(w1).callback(w1).param("1").next(workerWrapper3).build();WorkerWrapper<String, String> workerWrapper = new WorkerWrapper.Builder<String, String>().worker(w).callback(w).param("0").next(workerWrapper1, workerWrapper2).build();long now = SystemClock.now();System.out.println("begin-" + now);Async.beginWork(3100, workerWrapper);// Async.beginWork(2100, workerWrapper);System.out.println("end-" + SystemClock.now());System.err.println("cost-" + (SystemClock.now() - now));System.out.println(Async.getThreadCount());Async.shutDown();
如果觉得这样不符合左右的顺序,也可以用这种方式:
WorkerWrapper<String, String> workerWrapper = new WorkerWrapper.Builder<String, String>().worker(w).callback(w).param("0").build();WorkerWrapper<String, String> workerWrapper3 = new WorkerWrapper.Builder<String, String>().worker(w3).callback(w3).param("3").build();WorkerWrapper<String, String> workerWrapper2 = new WorkerWrapper.Builder<String, String>().worker(w2).callback(w2).param("2").depend(workerWrapper).next(workerWrapper3).build();WorkerWrapper<String, String> workerWrapper1 = new WorkerWrapper.Builder<String, String>().worker(w1).callback(w1).param("1").depend(workerWrapper).next(workerWrapper3).build();
三.复杂点的

在测试类里能找到,下图是执行结果。看时间戳,就知道执行的顺序。每个执行单元都是睡1秒。
- 依赖别的worker执行结果作为入参
 
可以从action的参数中根据wrapper的id获取任意一个执行单元的执行结果,但请注意执行顺序,如果尚未执行,则在调用WorkerResult.getResult()会得到null! 
