Java异步编程:CompletionStage接口详解
CompletionStage 接口分析
接口能力概述
CompletionStage 是 Java 8 引入的接口,用于表示异步计算的一个阶段,它提供了强大的异步编程能力:
- 链式异步操作:允许将一个异步操作的结果传递给下一个操作
- 组合操作:可以组合多个 CompletionStage
- 异常处理:提供对异步计算中异常的处理机制
- 多种执行方式:支持同步、默认异步和自定义执行器的异步执行
主要功能分类
1. 单阶段依赖操作
thenApply()
- 转换结果thenAccept()
- 消费结果thenRun()
- 执行无结果操作
2. 双阶段组合操作
thenCombine()
- 两个阶段都完成后合并结果thenAcceptBoth()
- 两个阶段都完成后消费结果runAfterBoth()
- 两个阶段都完成后执行操作
3. 任一阶段完成操作
applyToEither()
- 任一阶段完成后转换结果acceptEither()
- 任一阶段完成后消费结果runAfterEither()
- 任一阶段完成后执行操作
4. 异常处理
exceptionally()
- 异常时提供替代值handle()
- 无论成功或异常都处理whenComplete()
- 无论成功或异常都执行操作
5. 组合其他 CompletionStage
thenCompose()
- 扁平化嵌套的 CompletionStage
默认方法
从 Java 12 开始,CompletionStage 接口新增了一些默认方法,主要用于更灵活的异常处理:
-
exceptionallyAsync(Function<Throwable, ? extends T> fn)
- 异步处理异常的默认方法
-
exceptionallyAsync(Function<Throwable, ? extends T> fn, Executor executor)
- 使用指定执行器异步处理异常的默认方法
-
exceptionallyCompose(Function<Throwable, ? extends CompletionStage<T>> fn)
- 异常时返回新的 CompletionStage 的默认方法
-
exceptionallyComposeAsync(Function<Throwable, ? extends CompletionStage<T>> fn)
- 异步方式异常时返回新的 CompletionStage 的默认方法
-
exceptionallyComposeAsync(Function<Throwable, ? extends CompletionStage<T>> fn, Executor executor)
- 使用指定执行器异步方式异常时返回新的 CompletionStage 的默认方法
执行模式
大多数方法都有三种变体:
- 同步:基本方法(如
thenApply
) - 默认异步:带
Async
后缀(如thenApplyAsync
) - 自定义执行器异步:带
Async
后缀和 Executor 参数(如thenApplyAsync(fn, executor)
)
总结
CompletionStage 接口为 Java 异步编程提供了强大的构建块,允许开发者以声明式的方式组合异步操作,处理成功和失败情况,并控制操作的执行方式(同步或异步)。从 Java 12 开始,通过新增的默认方法进一步增强了异常处理的灵活性。
CompletionStage 默认方法实现分析
CompletionStage 接口从 Java 12 开始引入了一系列默认方法,主要增强了异常处理的灵活性。这些默认方法提供了开箱即用的实现,但它们的正确运行依赖于接口中其他基本方法的正确实现。
默认方法分类与实现分析
1. 异步异常处理 (exceptionallyAsync
)
public default CompletionStage<T> exceptionallyAsync(Function<Throwable, ? extends T> fn) {return handle((r, ex) -> (ex == null)? this: this.<T>handleAsync((r1, ex1) -> fn.apply(ex1))).thenCompose(Function.identity());
}
实现分析:
- 首先调用
handle
方法检查是否有异常 - 如果没有异常(
ex == null
),返回当前阶段本身 - 如果有异常,使用
handleAsync
异步执行异常处理函数 - 最后通过
thenCompose
扁平化结果
依赖的子类实现:
handle()
handleAsync()
thenCompose()
2. 带执行器的异步异常处理 (exceptionallyAsync
with Executor)
public default CompletionStage<T> exceptionallyAsync(Function<Throwable, ? extends T> fn, Executor executor) {return handle((r, ex) -> (ex == null)? this: this.<T>handleAsync((r1, ex1) -> fn.apply(ex1), executor)).thenCompose(Function.identity());
}
实现分析:
与上一个方法类似,但使用指定的 Executor
来执行异步操作
依赖的子类实现:
handle()
handleAsync(Executor)
thenCompose()
3. 异常组合处理 (exceptionallyCompose
)
public default CompletionStage<T> exceptionallyCompose(Function<Throwable, ? extends CompletionStage<T>> fn) {return handle((r, ex) -> (ex == null)? this: fn.apply(ex)).thenCompose(Function.identity());
}
实现分析:
- 检查是否有异常
- 无异常时返回当前阶段
- 有异常时调用函数生成新的 CompletionStage
- 使用
thenCompose
扁平化结果
依赖的子类实现:
handle()
thenCompose()
4. 异步异常组合处理 (exceptionallyComposeAsync
)
public default CompletionStage<T> exceptionallyComposeAsync(Function<Throwable, ? extends CompletionStage<T>> fn) {return handle((r, ex) -> (ex == null)? this: this.handleAsync((r1, ex1) -> fn.apply(ex1)).thenCompose(Function.identity())).thenCompose(Function.identity());
}
实现分析:
类似 exceptionallyCompose
,但使用异步方式处理异常
依赖的子类实现:
handle()
handleAsync()
thenCompose()
子类需要提供的内容
虽然这些是默认方法,但它们的正确运行依赖于接口中其他基本方法的正确实现。子类需要确保以下方法的正确实现:
-
基本处理方法:
handle()
handleAsync()
handleAsync(Executor)
-
组合方法:
thenCompose()
thenComposeAsync()
thenComposeAsync(Executor)
-
其他基础方法:
- 所有非默认的 CompletionStage 方法,因为默认方法构建在这些基础方法之上
实现注意事项
-
线程安全:子类实现必须保证线程安全,因为 CompletionStage 可能被多个线程访问
-
执行保证:子类必须确保异步方法(
...Async
)确实在另一个线程执行 -
异常传播:必须正确实现异常传播机制,确保异常能通过依赖链传递
-
完成状态:必须正确维护阶段的完成状态(正常完成/异常完成)
-
执行顺序:必须保证依赖操作的执行顺序符合接口规范
示例:自定义实现的关键点
如果创建自定义的 CompletionStage 实现,必须特别注意:
class MyCompletionStage<T> implements CompletionStage<T> {// 必须实现所有非默认方法@Overridepublic <U> CompletionStage<U> thenApply(Function<? super T,? extends U> fn) {// 实现转换逻辑}@Overridepublic <U> CompletionStage<U> handle(BiFunction<? super T, Throwable, ? extends U> fn) {// 实现处理逻辑}// 其他必要方法的实现...// 默认方法会自动继承,但依赖于上述方法的正确实现
}
CompletionStage 默认方法的使用场景
CompletionStage 接口的默认方法(特别是 Java 12 引入的异常处理方法)在以下场景中特别有用:
1. 异步异常处理场景
典型场景:当需要异步处理异常且不希望阻塞当前线程时
CompletionStage<String> stage = someAsyncOperation().exceptionallyAsync(ex -> {// 异步执行异常恢复逻辑log.error("Operation failed, using fallback", ex);return "fallback-value";});
优势:
- 异常处理不会阻塞调用线程
- 适合处理耗时的异常恢复逻辑(如远程调用备用服务)
2. 需要自定义线程池的异常处理
典型场景:当异常处理需要特定线程池资源时
ExecutorService recoveryExecutor = Executors.newFixedThreadPool(2);CompletionStage<String> stage = someAsyncOperation().exceptionallyAsync(ex -> {// 在专用线程池中执行恢复逻辑return callBackendServiceB();}, recoveryExecutor);
优势:
- 避免异常处理占用主业务线程池
- 可以为不同类型的恢复操作分配不同的线程资源
3. 需要返回新 CompletionStage 的异常恢复
典型场景:当异常发生时需要触发另一个异步操作来恢复
CompletionStage<String> stage = someAsyncOperation().exceptionallyCompose(ex -> {// 当主操作失败时,尝试备用方案return fallbackAsyncOperation();});
实际应用:
- 主数据库查询失败时尝试从缓存获取
- 主服务不可用时调用备用服务
4. 复杂的异常处理流水线
典型场景:需要多级异常恢复策略时
CompletionStage<String> stage = someAsyncOperation().exceptionallyComposeAsync(ex -> {// 第一级恢复:尝试本地备用方案return tryLocalRecovery();}).exceptionallyComposeAsync(ex -> {// 第二级恢复:尝试远程恢复return tryRemoteRecovery();}, remoteRecoveryExecutor).exceptionally(ex -> {// 最后兜底方案return "ultimate-fallback";});
优势:
- 构建多层次的弹性恢复策略
- 每级恢复可以使用不同的执行策略(同步/异步/特定线程池)
5. 与现有代码的集成
典型场景:当需要将异常处理函数封装为 CompletionStage 时
Function<Throwable, String> legacyRecovery = ex -> {// 传统的同步恢复逻辑return LegacyRecoveryService.recover(ex);
};// 将传统恢复逻辑适配为异步处理
CompletionStage<String> stage = someAsyncOperation().exceptionallyAsync(legacyRecovery);
优势:
- 将同步恢复逻辑自动提升为异步处理
- 无需修改原有恢复逻辑的实现
6. 需要保留堆栈信息的场景
典型场景:当异步操作链中需要保留原始异常信息时
CompletionStage<String> stage = someAsyncOperation().handleAsync((result, ex) -> {if (ex != null) {// 在此添加额外上下文信息throw new RecoveryException("Failed in async operation", ex);}return result;}).exceptionallyComposeAsync(ex -> {// 可以访问到包装后的异常信息RecoveryException re = (RecoveryException) ex.getCause();return recoveryWithContext(re.getContext());});
优势:
- 避免异步操作链中异常信息丢失
- 可以在不同阶段添加诊断上下文
何时选择默认方法 vs 基础方法
场景 | 使用默认方法 | 使用基础方法 |
---|---|---|
简单的同步异常处理 | - | exceptionally() |
异步异常处理 | exceptionallyAsync() | - |
需要控制异常处理线程 | exceptionallyAsync(fn, executor) | - |
异常时需要触发新异步操作 | exceptionallyCompose() | - |
简单的异常转换 | - | handle() |
需要同时处理成功和失败 | - | whenComplete() |
实际案例:服务降级策略
public CompletionStage<Response> handleRequest(Request request) {return primaryService.callAsync(request).exceptionallyComposeAsync(ex -> {if (ex instanceof TimeoutException) {// 超时快速失败,不尝试降级throw new ServiceException("Timeout", ex);}// 其他异常尝试降级return fallbackService.callAsync(request);}, fallbackExecutor).exceptionally(ex -> {// 记录最终失败metrics.recordFailure(ex);return Response.failure("Service unavailable");});
}
在这个案例中,我们:
- 首先尝试主服务调用
- 如果失败(非超时),异步尝试降级服务
- 使用专门的线程池执行降级逻辑
- 最后兜底记录指标并返回友好错误
这些默认方法让复杂的异步错误处理模式能够以声明式的方式简洁表达。
相关文章:
Java异步编程:CompletionStage接口详解
CompletionStage 接口分析 接口能力概述 CompletionStage 是 Java 8 引入的接口,用于表示异步计算的一个阶段,它提供了强大的异步编程能力: 链式异步操作:允许将一个异步操作的结果传递给下一个操作组合操作&a…...
Java后端接受前端数据的几种方法
在前后端分离的开发模式中,前端(Vue)与后端(Java)的数据交互有多种格式,下面详细介绍几种常见的格式以及后端对应的接收方式。 一、JSON 格式 前端传输 在 Vue 里,可借助 axios 把数据以 JSO…...
Oracle OCP认证的技术定位怎么样?
一、引言:Oracle OCP认证的技术定位 Oracle Certified Professional(OCP)认证是数据库领域含金量最高的国际认证之一,其核心价值在于培养具备企业级数据库全生命周期管理能力的专业人才。随着数字化转型加速,OCP认证…...
powershell7.5@.net环境@pwsh7.5在部分windows10系统下的运行问题
文章目录 powershell7.5及更高版本和.net 9解决方案 powershell7.5及更高版本和.net 9 相对较新的.Net 9版本在老一些的windows10系统上(比如内核版本号:10.0.19044.1288以及之前的),由于默认启用了CET,导致编译运行失败,需要自己在项目中添加关闭CET的配置语句才能够顺利编译…...

基于微信小程序的垃圾分类系统
博主介绍:java高级开发,从事互联网行业六年,熟悉各种主流语言,精通java、python、php、爬虫、web开发,已经做了六年的毕业设计程序开发,开发过上千套毕业设计程序,没有什么华丽的语言࿰…...
CSS3 渐变、阴影和遮罩的使用
全文目录: 开篇语**前言****1. CSS3 渐变 (Gradient)****1.1 线性渐变 (linear-gradient)****1.2 径向渐变 (radial-gradient)** **2. CSS3 阴影 (Shadow)****2.1 盒子阴影 (box-shadow)****2.2 文本阴影 (text-shadow)** **3. CSS3 遮罩 (Mask)****3.1 基本遮罩 (m…...
Spring Boot 全局配置文件优先级
好的,Spring Boot的全局配置文件优先级是一个非常重要的概念,它决定了在不同位置的同名配置属性以哪个为准。 Spring Boot 全局配置文件优先级核心知识点 📌 文件格式优先级: 在同一目录下,如果同时存在 application.properties 和…...

流媒体基础解析:视频清晰度的关键因素
在视频处理的过程中,编码解码及码率是影响视频清晰度的关键因素。今天,我们将深入探讨这些概念,并解析它们如何共同作用于视频质量。 编码解码概述 编码,简单来说,就是压缩。视频编码的目的是将原始视频数据压缩成较…...

grid网格布局
使用flex布局的痛点 如果使用justify-content: space-between;让子元素两端对齐,自动分配中间间距,假设一行4个,如果每一行都是4的倍数那没任何问题,但如果最后一行是2、3个的时候就会出现下面的状况: /* flex布局 两…...
C#数字金额转中文大写金额:代码解析
C#数字金额转中文大写金额:代码解析 在金融相关的业务场景中,我们常常需要将数字金额转换为中文大写金额,以避免金额被篡改,增加金额的准确性和安全性。本文将深入解析一段 C# 代码,这段代码通过巧妙的设计࿰…...

Vehicle HAL(2)--Vehicle HAL 的启动
目录 1. VehicleService-main 函数分析 2. 构建EmulatedVehicleHal 2.1 EmulatedVehicleHal::EmulatedVehicleHal(xxx) 2.2 EmulatedVehicleHal::initStaticConfig() 2.3 EmulatedVehicleHal::onPropertyValue() 3. 构建VehicleEmulator 4. 构建VehicleHalManager (1)初…...
JS中的函数防抖和节流:提升性能的关键技术
在JavaScript开发中,函数防抖和节流是两种常用的优化技术,用于处理那些可能会被频繁触发的事件,如resize、scroll、mousemove等。本文将详细介绍函数防抖和节流的概念、实现方法以及它们之间的区别。 一、什么是函数防抖和节流? …...
Android Compose开发架构选择指南:单Activity vs 多Activity
简介 掌握Jetpack Compose的Activity架构选择,是构建高性能、易维护Android应用的关键一步。在2025年的Android开发领域,随着Jetpack Compose的成熟和Android系统对多窗口模式的支持,开发者面临架构选择时需要更加全面地考虑各种因素。本文将深入探讨单Activity架构和多Act…...
【Netty系列】Reactor 模式 1
目录 一、Reactor 模式的核心思想 二、Netty 中的 Reactor 模式实现 1. 服务端代码示例 2. 处理请求的 Handler 三、运行流程解析(结合 Reactor 模式) 四、关键点说明 五、与传统模型的对比 六、总结 Reactor 模式是 Netty 高性能的核心设计思想…...
vue3 el-input type=“textarea“ 字体样式 及高度设置
在Vue 3中,如果你使用的是Element Plus库中的<el-input>组件作为文本域(type"textarea"),你可以通过几种方式来设置字体样式和高度。 1. 直接在<el-input>组件上使用style属性 你可以直接在<el-input&…...
并发解析hea,转为pdf格式
由于每次解析一个heap需要时间有点久,就写了一个自动解析程pdf的一个脚本。 down_lib.sh是需要自己写的哦,主要是用于下载自己所需程序的库,用于解析heap。 #!/bin/bash# 优化版通用解析脚本(并发加速):批…...

【C语言】详解 指针
前言: 在学习指针前,通过比喻的方法,让大家知道指针的作用。 想象一下,你在一栋巨大的图书馆里找一本书。如果没有书架编号和目录,这几乎是不可能完成的任务。 在 C 语言中,指针就像是图书馆的索引系统&…...

RabbitMQ仲裁队列高可用架构解析
#作者:闫乾苓 文章目录 概述工作原理1.节点之间的交互2.消息复制3.共识机制4.选举领导者5.消息持久化6.自动故障转移 集群环境节点管理仲裁队列增加集群节点重新平衡仲裁队列leader所在节点仲裁队列减少集群节点 副本管理add_member 在给定节点上添加仲裁队列成员&…...
刚出炉热乎的。UniApp X 封装 uni.request
HBuilder X v4.66 当前最新版本 由于 uniapp x 使用的是自己包装的 ts 语言 uts。目前语言还没有稳定下来,各种不支持 ts 各种报错各种不兼容问题。我一个个问题调通的,代码如下: 封装方法 // my-app/utils/request.uts const UNI_APP_BASE…...

Apache Kafka 实现原理深度解析:生产、存储与消费全流程
Apache Kafka 实现原理深度解析:生产、存储与消费全流程 引言 Apache Kafka 作为分布式流处理平台的核心,其高吞吐、低延迟、持久化存储的设计使其成为现代数据管道的事实标准。本文将从消息生产、持久化存储、消息消费三个阶段拆解 Kafka 的核心实现原…...

Python 训练营打卡 Day 41
简单CNN 一、数据预处理 在图像数据预处理环节,为提升数据多样性,可采用数据增强(数据增广)策略。该策略通常不改变单次训练的样本总数,而是通过对现有图像进行多样化变换,使每次训练输入的样本呈现更丰富…...

leetcode付费题 353. 贪吃蛇游戏解题思路
贪吃蛇游戏试玩:https://patorjk.com/games/snake/ 问题描述 设计一个贪吃蛇游戏,要求实现以下功能: 初始化游戏:给定网格宽度、高度和食物位置序列移动操作:根据指令(上、下、左、右)移动蛇头规则: 蛇头碰到边界或自身身体时游戏结束(返回-1)吃到食物时蛇身长度增加…...

CCPC dongbei 2025 I
题目链接:https://codeforces.com/gym/105924 题目背景: 给定一个二分图,左图编号 1 ~ n,右图 n 1 ~ 2n,左图的每个城市都会与右图的某个城市犯冲(每个城市都只与一个城市犯冲),除…...

系统性学习C语言-第十三讲-深入理解指针(3)
系统性学习C语言-第十三讲-深入理解指针(3) 1. 数组名的理解2. 使用指针访问数组3. ⼀维数组传参的本质4. 冒泡排序5. ⼆级指针 6. 指针数组7. 指针数组模拟二维数组 1. 数组名的理解 在上⼀个章节我们在使用指针访问数组的内容时,有这样的代…...
代理模式核心概念
代理模式核心概念 代理模式是一种结构型设计模式,通过创建一个代理对象来控制对原始对象的访问。主要分为两类: 一、静态代理 (Static Proxy) 定义:在编译期确定代理关系的模式,代理类和目标类都需要实现相同的接口。 核心特点…...
uni-app学习笔记十五-vue3页面生命周期(二)
onShow:用于监听页面显示,页面每次出现在屏幕上都触发,包括从下级页面点返回露出当前页面; onHide:监听页面隐藏,当离开当前页面时触发。 示例代码: <template><view>姓名:{{nam…...

贪心算法实战篇2
文章目录 前言序列问题摆动序列单调递增的数字 贪心解决股票问题买卖股票的最佳时机II 两个维度权衡问题分发糖果根据身高重建队列 前言 今天继续带大家进行贪心算法的实战篇2,本章注意来解答一些运用贪心算法的中等的问题,大家好好体会,怎么…...

Java 大视界 -- Java 大数据机器学习模型在元宇宙虚拟场景智能交互中的关键技术(239)
💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也…...
Flask中关于app.url_map属性的用法
目录 一、app.url_map 是什么? 二、可以查看哪些信息? 三、示例:打印所有路由 四、结合 url_for() 使用 五、常见用途场景 六、结合 Flask CLI 使用 总结 app.url_map 是 Flask 中非常重要的一个属性,用于查看或操作整个应用的 URL 路由映射表(routing map)。它展…...

高速串行接口
1.网口设计方案 上图中给出了两种网口设计方案,最上面是传统设计方式,下面是利用GT作为PHY层的设计,然后FPGA中设计协议层和MAC层。 2.SRIO SRIO的本地操作和远程操作 3.其他高速接口 srio rapid io aurora8b10b aurora64b66b pcie s…...