异步编程的两种场景

在异步编程中,回调函数通常服务于两种不同场景:

  1. 一次性资源获取:等待异步操作完成并返回结果。
  2. 持续事件通知。监听并响应多个状态变更。

Kotlin为这两种场景提供了解决方案:使用挂起函数简化一次性资源获取,使用流处理持续事件通知。关于事件流处理方案详见《将listener转换为事件流》一文。本文聚焦第一种场景:如何简化异步资源获取操作。

异步编程的挑战

异步获取资源接口一般会提供两个状态回调函数:

interface ResourceStateListener {fun onReady(resource: Resource)fun onGone(error: Throwable)
}

有些复杂接口可能提供更多回调函数:

interface ComplexResourceStateListener {fun onOpen(resource: Resource)fun onReady(resource: Resource)fun onError(error: Throwable)fun onConfigureFailure(errorCode: Int)fun onClose()
}

客户代码的核心需求是获取可用资源。根据奥卡姆剃刀“如无必要,勿增实体”的原则,可以将所有资源不可用状态onError/onConfigureFailure/onClose合并成一个:onGone。考虑到onOpen事件只和资源清理有关,不执行业务操作。因此我们真正要关心的只有onReady和onGone。

传统回调的困境

根据上面的例子,我们可以得到代码:

fun openResource(resId: String, listener: ResourceStateListener) { ... }val resId = "1"
openResource(resId, object: ResourceStateListener {override fun onReady(resource: Resource) { ... }override fun onGone(error: Throwable) { ... }
})

传统回调模式存在以下问题:

  1. 代码逻辑分散。资源申请逻辑、使用资源逻辑、错误处理逻辑、资源清理逻辑分割在不同上下文中,代码难以追踪资源状态变化的完整路径。
  2. 状态管理困难。客户代码需要引入额外的变量来在回调函数之间传递资源对象或状态信息,代码复杂度,容易出错。
  3. 容易泄露资源。由于代码分散,难以追踪状态变化的完整路径,很难正确释放资源。容易存在资源泄露或重复释放。
  4. 可读性和可维护性差。回调模式无法满足结构化编程的“单一入口,单一出口”要求,代码难以阅读、理解和修改。

从结构化编程的角度来看,传统回调模式将申请资源代码、使用资源代码和异常处理代码分散在不同的上下文中,无法形成单一入口单一出口的逻辑结构,几乎是现代版的goto变种。

以同步形式编写异步代码

重新观察获取资源的过程:

  1. 程序向系统提交资源申请。系统以异步方式处理申请。
  2. 程序等待系统通知申请结果。
  3. 程序根据结果执行操作。

考虑到Kotlin挂起函数和协程非常适合等待场景,我们可以构造一个挂起函数,发起异步请求后函数挂起,等待回调函数唤醒。对于onReady事件,通过resume唤醒,进入使用资源逻辑。对于onGone事件,通过resumeWithException唤醒,进入错误处理逻辑。同时利用结构化并发特性,在协程退出时清理资源。

suspend fun openResource(resId: String): Resource = suspendCancellableCoroutine { cont ->var internalRes: Resource? = nullval listener = object : ComplexResourceStateListener {override fun onOpen(resource: Resource) {internalRes = resource // 保存底层资源对象引用,必要时手动释放。}override fun onReady(resource: Resource) {if (cont.isActive) {cont.resume(resource) // 资源就绪,唤醒协程。}}override fun onError(error: Throwable) {if (cont.isActive) {cont.resumeWithException(ResourceUnavailableException("Resource error", error))}}override fun onConfigureFailure(errorCode: Int) {if (cont.isActive) {cont.resumeWithException(ResourceUnavailableException("Configuration failed: $errorCode"))}}override fun onClose() {if (cont.isActive) {cont.resumeWithException(ResourceUnavailableException("Resource closed prematurely"))}}}// 发起异步请求。resource.openResource(resId, listener)// 设置资源清理操作。cont.invokeOnCancellation {// 移除监听器,避免后续无效回调干扰和内存泄漏。removeStateListener(listener)// 释放底层资源对象。internalRes?.release()internalRes = null}// 协程在此挂起,等待唤醒。
}// 资源不可用异常
class ResourceUnavailableException(message: String, cause: Throwable? = null) : Exception(message, cause)

封装成挂起函数之后,就可以在协程中以同步的形式编码。

try {val res = openResource(resId)useResource(res)
} catch (e: Exception) {handleError(e)
}

可以看到,使用协程封装回调函数拥有以下优势:

  1. 同步风格代码。使用线性代码结构来表达业务逻辑,代码简洁、意图清晰、可读性良好。
  2. 逻辑完整。资源申请、使用、错误处理逻辑集中在同一个上下文中,理解和维护成本低。
  3. 资源安全。利用协程的invokeOnCancellation和结构化并发特性,​保证资源安全释放,减少资源泄漏风险。
  4. 错误处理简单。所有导致资源不可用的底层错误统一转换成语义清晰的单一异常,简化客户代码错误处理逻辑。
  5. 支持取消。利用协程的取消机制可以取消操作,释放资源。
  6. 接口简单。对客户代码屏蔽了复杂的底层细节,只发布一个简单的挂起函数,提高接口的易用性和语义清晰度。

通过使用协程进行封装,我们将原本支离破碎、难以管理的异步代码转变成结构清晰、资源安全、易于编写和维护的“同步”代码。即提升了开发效率,也大幅增强了程序健壮性。

参考资料

  • 协程指南
  • 将listener转换为事件流
  • 只崩溃软件
  • 我对续体传递风格CPS的理解
  • 结构化并发
  • 结构化并发(2)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/87983.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/87983.shtml
英文地址,请注明出处:http://en.pswp.cn/web/87983.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ABP VNext + Cosmos DB Change Feed:搭建实时数据变更流服务

ABP VNext Cosmos DB Change Feed:搭建实时数据变更流服务 🚀 📚 目录ABP VNext Cosmos DB Change Feed:搭建实时数据变更流服务 🚀TL;DR ✨🚀1. 环境与依赖 🏗️2. 服务注册与依赖注入 &…

STM32-定时器

定时器:有4个独立通道:输入捕获;输出比较PWM生成;单脉冲模式输出;可通外部信号控制定时器(TIMx-ETR);支持针对定时的增量(正交)编码器、霍尔传感器电路通用定…

Windows Server 2019--职业技能大赛B模块Windows服务器配置样题

一、赛题说明 (一)竞赛介绍 请详细阅读网络拓扑图,为所有计算机修改默认防火墙以便允许ICMP和相应的流量,不允许直接关闭主机的防火墙。除了CD-ROM/HDD驱动器,请不要修改虚拟机本身的硬件设置。 (二&…

vue3+Echarts实现立体柱状图

Echarts柱状图中文网:https://echarts.apache.org/examples/zh/index.html#chart-type-bar 效果展示: 主要实现过程是三部分的组合,最上面是一个椭圆,中间是正常的柱子,下方再加上一个椭圆,就出来立体的效…

【UE5】虚幻引擎小百科

一、类名前面的大写字母的含义是什么UE5常见前缀分类表前缀含义实例用于AActorACharacter,AWeaponBase可放入世界中的对象(有位置、可碰撞等)UUObject派生类UUserWidget,UWeaponComponent引擎对象、逻辑模块,不具备Tra…

【Linux系统】vim编辑器 | 编译器gcc/g++ | make/Makefile

1. vim编辑器一、历史发展与Vim vs Vi的区别起源与演进Vi(1976年) :由Bill Joy开发,嵌入BSD Unix系统,是首个面向屏幕的文本编辑器,但功能有限(如无多级撤销)。Vim(1991年…

国产飞腾主板,赋能网络安全防御硬手段

​ 当前,网络安全形势严峻,网络攻击手段不断翻新,从数据泄露到电脑中毒,企业、机构乃至国家的数字资产都面临着巨大风险。在此背景下,国产硬件技术的突破对筑牢网络安全防线意义重大。 高能计算机基于市场需求&#…

Spring AI 概述与架构设计

目录一、前言二、简介三、核心能力概览四、理解模块架构图五、模型适配能力六、最小应用示例七、与传统 LLM 调用相比八、总结九、参考一、前言 在 AI 正以前所未有的速度“下沉”到各类系统与业务的当下,Spring 官方推出的 Spring AI 项目,为 Java 开发…

UI前端与数字孪生融合新领域:智慧环保的污染源监测与治理

hello宝子们...我们是艾斯视觉擅长ui设计、前端开发、数字孪生、大数据、三维建模、三维动画10年经验!希望我的分享能帮助到您!如需帮助可以评论关注私信我们一起探讨!致敬感谢感恩!一、引言:数字孪生重构智慧环保的技术范式在环境污染治理压力持续增大的背景下&…

【go/wails】wails入门系列(一)环境安装与demo

文章目录说在前面go安装nodejs安装wails创建项目运行说在前面 操作系统:win11go版本:1.24.4nodejs版本:v22.16.0wails版本:v2.10.1 go安装 官网 这里 下载安装即可 nodejs 官网 这里 下载安装即可 安装wails 设置go国内代理g…

linux qt 使用log4cpp库

一、日志库下载 下载地址:https://log4cpp.sourceforge.net/二、日志库解压,编译 1.将文件夹解压出来2.进入文件夹内部,打开终端3.终端中依次输入以下命令 mkdir build ./configure --prefix$(pwd)/build make make install 一般来说不会报错…

探索阿里云Data Integration:数据同步的魔法工具

引言在当今数字化时代,数据已成为企业的核心资产,如同企业发展的 “燃料”,驱动着业务的增长与创新。从用户行为数据到业务运营数据,从市场趋势数据到供应链数据,每一个数据点都蕴含着巨大的价值,能够为企业…

【Java面试】Redis的poll函数epoll函数区别?

Redis 在选择 poll 和 epoll 时主要基于性能需求、连接规模、操作系统支持等因素。以下是具体场景的对比与选择建议:1. 何时使用 poll 函数?适用场景: 跨平台兼容性需求:poll 在几乎所有操作系统(如 Windows、BSD、Lin…

RPC--RPCHandler的实现

在RPC框架中,Handler用于接收RpcRequest,经过处理后返回RpcResponseSlf4jpublic class RpcRequestHandler {private final ServiceProvider serviceProvider;//获取一个单例模式的服务提供类public RpcRequestHandler() {serviceProvider SingletonFact…

C#读取文件夹和文件列表:全面指南

C#读取文件夹和文件列表:全面指南 在 C# 开发中,经常需要获取文件夹中的文件列表或子文件夹结构,例如文件管理器、批量处理工具、备份程序等场景。本文将详细介绍 C# 中读取文件夹和文件列表的各种方法,包括基础操作、递归遍历、过…

从小白到进阶:解锁linux与c语言高级编程知识点嵌入式开发的任督二脉(1)

【硬核揭秘】Linux与C高级编程:从入门到精通,你的全栈之路!第一部分:初识Linux与环境搭建,玩转软件包管理——嵌入式开发的第一道“坎”嘿,各位C语言的“卷王”们!你可能已经习惯了在Windows或m…

.net开源库SignalR

.NET开源库SignalR:打造实时Web应用的利器 在当今的Web开发领域,实时性已经成为了许多应用的核心需求。无论是实时聊天、实时数据监控还是实时游戏,都需要服务器能够及时地将数据推送给客户端。而.NET开源库SignalR,正是满足这一…

SQL Server不同场景批量插入数据的方式详解

INSERT INTO...VALUES多行语法 该方法适用于单次插入少量数据(通常<1000行),语法简洁直观。示例: INSERT INTO Employees (EmployeeID, Name, Department) VALUES (101, Zhang San, IT),(102, Li Si, HR),(103, Wang Wu, Finance)优点:语法简单易理解,适合开发测试环…

Day08-Flask 或 Django 简介:构建 Web 应用程序

Flask 或 Django 简介&#xff1a;构建 Web 应用程序 网络开发领域提供了丰富的工具和框架&#xff0c;而 Python 作为一门多功能的语言&#xff0c;在构建健壮且可扩展的 Web 应用方面脱颖而出。本课程将作为你使用 Python 进行 Web 开发的入门指南&#xff0c;特别聚焦于两个…

k8s多集群管理中的联邦和舰队如何理解?

在 Kubernetes 多集群管理中&#xff0c;联邦&#xff08;Federation&#xff09;和舰队&#xff08;Fleet&#xff09;是两种不同的方法&#xff0c;用于管理和协调多个 Kubernetes 集群。下面是对这两种方法的详细解释&#xff1a; 联邦&#xff08;Federation&#xff09; K…