【多线程开发 3】从源码到实践CompletableFuture
2024年5月22日
本文将从以下几个点讲解CompletableFuture:
-
前身
-
是什么?
-
可以用来做什么?
-
源码原理
-
实战
-
其他包装类
前身
CompletableFuture的依赖如图所示:
所以我打算从Runnable、Future、RunnableFuture、FutureTask讲起
Runnable
Runnable就是一个带着执行方法的接口,通过实现Runnable可以实现开始一个线程执行对应的无返回值的操作。
Future
Future保存异步计算的结果,可以在我们执行任务时去做其他工作,体现了解耦的思想。Future 可以对于具体的Runnable或者Callable任务的执行结果进行取消、查询是否完成、获取结果。必要时可以通过get方法获取执行结果,该方法会阻塞直到任务返回结果。
其实本质上就是对Runnable或者Callable任务做了一层异步方法封装
RunnableFuture
RunnableFuture实际上只是Runnable接口和Future接口的总和,使用较少。
FutureTask
FutureTask是Future接口的一个唯一实现类。
FutureTask可以对Task做一个异步执行,通过FutureTask的构造函数可以将你想进行异步操作的Task封装进去。FutureTask是个适配器类,作为一个转换器,将Runnable和Callable适配,适配器类是适配器模式的核心,它通过实现RunnableFuture和组合Callable使得两者产生关系。
CompletableFuture可以做什么?
如果只使用Future类可以进行简单的异步执行,但是使用Future的get方法会导致主进程阻塞,并且也无法做到多个异步任务之间的依赖关系,并且有时候异步任务报错不会输出堆栈,Future任务也不会触发后续操作,没有提供函数式编程,不过以上CompletableFuture都可以帮助我们实现。
实战
以下是一些常用API,需要注意的是CompletableFuture可以使用默认线程池或者是自定义线程池,默认线程池是ForkJoinPool
@Test
public void test1() throws ExecutionException, InterruptedException, java.util.concurrent.ExecutionException {
CompletableFuture<String> base = new CompletableFuture<>();
CompletableFuture<String> future = base.thenApply(s -> s + " 2").thenApply(s -> s + " 3");
base.complete("1");
System.out.println(future.get());
}
@Test
public void test2() throws ExecutionException, InterruptedException, java.util.concurrent.ExecutionException {
CompletableFuture<String> base = new CompletableFuture<>();
CompletableFuture