电话
400 9058 355
CompletableFuture 解决了传统 Future 无法链式处理、组合任务、非阻塞回调、异常统一处理等硬伤,支持 thenApply/thenCompose/anyOf/allOf 等编排能力,并提供 handle/whenComplete/exceptionally 等语义明确的回调机制。
Java 5 引入的 Future 只能被动等待结果,无法链式处理、无法组合多个异步任务、也不能方便地处理异常或 fallback。它像一张单程票——你交出任务,只能等 get() 阻塞拿结果,中间没任何干预能力。
而 CompletableFuture(JDK 8 加入)补全了这整条异步流水线:支持非阻塞回调、任务编排、异常传播、超时控制、手动完成——本质是把“异步操作”真正变成可组合、可观察、可中断的编程单元。
Future.get() 一调就卡主线程;CompletableFuture 的 thenApply
thenAccept 等方法全部异步触发,不阻塞调用线程thenCompose 可自然平铺链式依赖CompletableFuture.anyOf() 直接返回新 CompletableFuture;想等全部完成?用 allOf()
completeExceptionally(new RuntimeException()) 可在任意时刻主动失败,比靠 get() 抛异常更可控二者都能创建已完成的 CompletableFuture,但语义和线程模型完全不同。
supplyAsync(() -> doWork()) 会把 doWork() 提交到默认的 ForkJoinPool.commonPool()(或指定线程池),适合真正耗时的计算或 I/O;而 new CompletableFuture().complete(value) 是立即完成,不涉及线程切换,适合模拟结果、测试或快速返回缓存值。
supplyAsync 时注意:默认线程池不支持 IO 密集型任务长时间阻塞,容易拖垮整个 commonPool;IO 类操作建议传自定义线程池,比如 supplyAsync(() -> callHttp(), httpClientPool)
complete() 和 completeExceptionally() 只能调用一次,重复调用无效;而 supplyAsync 每次都新建一个异步任务doWork() 是纯内存计算且极快(如 JSON 解析小对象),直接 completedFuture(value) 更轻量,避免线程调度开销它们都用于响应完成事件,但职责边界清晰,混用容易导致逻辑错乱或异常丢失。
handle((result, ex) -> { ... }):无论成功或失败都会执行,必须返回值(可为 null),用于统一转换结果或兜底处理;ex 为 null 表示正常完成whenComplete((result, ex) -> { ... }):也是无论成败都执行,但不改变原始结果,返回仍是原 CompletableFuture;适合打日志、清理资源,不能用来 fallbackexceptionally(ex -> fallbackValue):仅在异常时触发,且只接收 Throwable,返回替代值;它不会吞掉异常——如果 fallback 也抛异常,上游仍能捕获常见错误:用 whenComplete 做 fallback(它不返回新值),或在 handle 里抛出未检查异常却不处理(会导致下游 get() 抛 CompletionException)。
allOf() 只保证全部完成,并不聚合结果,返回类型是 CompletableFuture ——这是最容易踩的坑。它本身不持有各个子任务的返回值。
Stream 收集所有 CompletableFuture,再调用 allOf() 等待完成,最后用 join() 或 get() 逐个取值CompletableFuturef1 = CompletableFuture.supplyAsync(() -> "a"); CompletableFuture f2 = CompletableFuture.supplyAsync(() -> 42); CompletableFuture all = CompletableFuture.allOf(f1, f2); all.join(); // 等待全部完成 String s = f1.join(); // 再单独取 Integer i = f2.join();
stream.map(CompletableFuture::join).collect(...),或借助第三方库如 CompletableFutures.allAsList()(Guava)别指望 allOf() 自动打包成 List —— 它的设计哲学就是“只管完成信号,不管数据”,这点和 RxJava 的 zip 或 combineLatest 有本质区别。
邮箱:8955556@qq.com
Q Q:8955556
本文详解如何将Go官方present工具(用于生成HTML5...
PySNMP在不同版本中对SNMP错误状态(errorSta...
time.Sleep仅阻塞当前goroutine,其他gor...
PHPfopen()创建含特殊符号的文件名失败主因是操作系统...
WooCommerce中通过代码为分组产品动态聚合子商品的属...
io.ReadFull返回io.ErrUnexpectedE...
本文详解Yii2中控制器向视图传递ActiveRecord数...
本文详解为何通过wp_set_object_terms()为...
Pytest中使用@mock.patch类装饰器会导致补丁泄...
带缓冲的channel是并发安全的FIFO队列;make(c...