使用協程簡化異步資源獲取操作

異步編程的兩種場景

在異步編程中,回調函數通常服務于兩種不同場景:

  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…