CompletableFuture详解

介绍:

Java 8中新增的java.util.concurrent.CompletableFuture类是一个用于异步编程的工具类,它提供了一种方便的方法来处理异步任务的结果。CompletableFuture类可以将异步任务和回调函数组合在一起,实现异步任务的执行和结果的处理,同时也提供了一些方便的方法来处理异常和并发问题。

类结构:

public class CompletableFuture<T> implements Future<T>, CompletionStage<T> {
    // ...
}

异步执行:

CompletableFuture类提供了两种方式来执行异步任务:supplyAsync()和runAsync()。这些方法会在ForkJoinPool中执行异步任务,可以指定Executor来执行任务。

supplyAsync(Supplier<U> supplier):用于执行一个有返回值的异步任务,接收一个Supplier<U>类型的参数,返回一个CompletableFuture<U>对象。该方法会使用默认的线程池执行任务。

supplyAsync(Supplier<U> supplier, Executor executor):用于执行一个有返回值的异步任务,接收一个Supplier<U>类型的参数和一个指定的Executor对象,返回一个CompletableFuture<U>对象。该方法会使用指定的线程池执行任务。

runAsync(Runnable runnable):用于执行一个没有返回值的异步任务,接收一个Runnable类型的参数,返回一个CompletableFuture<Void>对象。该方法会使用默认的线程池执行任务。

runAsync(Runnable runnable, Executor executor):用于执行一个没有返回值的异步任务,接收一个Runnable类型的参数和一个指定的Executor对象,返回一个CompletableFuture<Void>对象。该方法会使用指定的线程池执行任务。

回调函数:

CompletableFuture类提供了一系列的thenXXX()方法,用于在异步任务完成后执行回调函数。这些方法可以串行执行、并行执行、组合执行等,能够满足各种不同的需求

thenApply(Function<? super T, ? extends U> fn):在异步任务完成后执行一个操作,并返回一个新的CompletableFuture对象,该方法会在当前线程中执行回调函数。

thenApplyAsync(Function<? super T, ? extends U> fn):在异步任务完成后执行一个操作,并返回一个新的CompletableFuture对象,该方法会使用ForkJoinPool中的线程执行回调函数。

thenApplyAsync(Function<? super T, ? extends U> fn, Executor executor):在异步任务完成后执行一个操作,并返回一个新的CompletableFuture对象,该方法会使用指定的Executor执行回调函数。

thenAccept(Consumer<? super T> action):在异步任务完成后执行一个操作,不返回任何结果,该方法会在当前线程中执行回调函数。

thenAcceptAsync(Consumer<? super T> action):在异步任务完成后执行一个操作,不返回任何结果,该方法会使用ForkJoinPool中的线程执行回调函数。

thenAcceptAsync(Consumer<? super T> action, Executor executor):在异步任务完成后执行一个操作,不返回任何结果,该方法会使用指定的Executor执行回调函数。

thenRun(Runnable action):在异步任务完成后执行一个操作,不接受任何参数和返回值,该方法会在当前线程中执行回调函数。

thenRunAsync(Runnable action):在异步任务完成后执行一个操作,不接受任何参数和返回值,该方法会使用ForkJoinPool中的线程执行回调函数。

thenRunAsync(Runnable action, Executor executor):在异步任务完成后执行一个操作,不接受任何参数和返回值,该方法会使用指定的Executor执行回调函数。

thenCompose(Function<? super T, ? extends CompletionStage<U>> fn):串行执行两个异步任务,并返回一个新的CompletableFuture对象,该方法会在当前线程中执行回调函数。

thenComposeAsync(Function<? super T, ? extends CompletionStage<U>> fn):串行执行两个异步任务,并返回一个新的CompletableFuture对象,该方法会使用ForkJoinPool中的线程执行回调函数。

thenComposeAsync(Function<? super T, ? extends CompletionStage<U>> fn, Executor executor):串行执行两个异步任务,并返回一个新的CompletableFuture对象,该方法会使用指定的Executor执行回调函数。

异常处理:

CompletableFuture类提供了一系列方法来处理异常,包括exceptionally()、handle()和whenComplete()等。这些方法可以处理异步任务的异常,保证程序的稳定性

exceptionally(Function<Throwable, ? extends T> fn):如果异步任务出现异常,则执行指定的操作,并返回一个新的CompletableFuture对象,该方法会在当前线程中执行回调函数。

handle(BiFunction<? super T, Throwable, ? extends U> fn):如果异步任务出现异常,则执行指定的操作,并返回一个新的CompletableFuture对象,该方法会在当前线程中执行回调函数。

whenComplete(BiConsumer<? super T, ? super Throwable> action):在异步任务完成后,无论是否出现异常都会执行指定的操作,并返回一个新的CompletableFuture对象,该方法会在当前线程中执行回调函数。

这三个方法的主要区别在于异常处理和返回值类型。exceptionally()方法会处理异常并返回一个新的CompletableFuture对象,但是返回值类型仍然是原来的类型T。handle()方法也可以处理异常,但可以返回一个新的类型U。whenComplete()方法不会处理异常,但可以执行指定的操作,并返回一个新的CompletableFuture对象。

并发问题:

CompletableFuture类提供了一些方便的方法来处理并发问题,包括allOf()、anyOf()等方法。这些方法可以并行执行多个异步任务,并等待任务完成。

allOf(CompletableFuture<?>... cfs):接收一个或多个CompletableFuture对象作为参数,返回一个新的CompletableFuture<Void>对象。这个方法会等待所有传入的CompletableFuture对象都完成(无论是正常完成还是异常完成)后,才会完成返回的CompletableFuture<Void>对象。

anyOf(CompletableFuture<?>... cfs):接收一个或多个CompletableFuture对象作为参数,返回一个新的CompletableFuture<Object>对象。这个方法会等待任意一个传入的CompletableFuture对象完成(无论是正常完成还是异常完成),然后返回一个包含已完成任务结果的CompletableFuture<Object>对象。

简单的代码示例:

CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
    // 异步执行一个任务
    return "Hello, world!";
}).thenApply(result -> {
    // 在异步任务完成后执行一个操作
    return result.toUpperCase();
}).thenApply(result -> {
    // 在异步任务完成后执行另一个操作
    return "Result: " + result;
});

// 获取异步任务的结果
String result = future.get();
System.out.println(result);

性能和风险:

使用CompletableFuture类可以实现异步编程,避免了阻塞等待的情况,可以提高程序的性能和响应速度。但是,在使用CompletableFuture类时,需要注意以下几点风险:

可能会引起死锁:当使用thenCompose()方法时,需要小心处理任务之间的依赖关系,避免出现死锁的情况。
可能会导致线程池满载:当并发执行大量的异步任务时,可能会导致线程池满载,从而影响程序的性能和稳定性。
可能会引发异常:异步任务的执行可能会引发异常,需要小心处理异常,以避免影响程序的稳定性。

对应措施:

合理使用线程池:使用CompletableFuture类时,需要合理使用线程池,避免线程池满载的情况。可以根据任务的类型和执行时间等因素,动态调整线程池的大小。
小心处理任务依赖关系:当使用thenCompose()方法时,需要小心处理任务之间的依赖关系,避免出现死锁的情况。可以使用thenApply()方法来代替thenCompose()方法,避免出现死锁的风险。
小心处理异常:异步任务的执行可能会引发异常,需要小心处理异常,以避免影响程序的稳定性。可以使用exceptionally()方法来处理异常,避免异常的抛出影响程序的执行。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值