Kotlin 协程高级用法之 NonCancellable

核心前提:
NonCancellable = 临时屏蔽取消信号 ,让代码块原子执行不中断可用于任意代码位置 ,没有强制必须写在 finally 里。


用法1:官方推荐用法 → finally 块中(资源清理/收尾)

适用场景

协程无论成功/失败/取消 ,都必须执行的资源释放操作 (关流、关连接、释放锁);

因为 finally 里的挂起函数 会被取消中断,所以必须用 NonCancellable 包裹。

Android 实战代码

kotlin 复制代码
import kotlinx.coroutines.*
import androidx.lifecycle.lifecycleScope

// Android 页面中(Activity/Fragment)
lifecycleScope.launch(Dispatchers.IO) {
    var networkStream: NetworkStream? = null
    try {
        // ========== 可取消区域 ==========
        // 耗时网络/文件操作,cancel() 会立即中断
        networkStream = openNetworkStream()
        readStreamData(networkStream)
        delay(1000) // 挂起点,可被取消
    } finally {
        // ========== 必执行区域:协程任何结束方式都会进入 ==========
        // 官方规范:finally 中执行挂起收尾,必须用 NonCancellable
        withContext(NonCancellable) {
            // 纯资源清理,无业务成功逻辑
            networkStream?.close() // 挂起函数,不被取消中断
            releaseWorkerLock()
            log("资源已安全释放")
        }
    }
}

用法2:项目实战用法 → 业务成功分支中(原子提交)

适用场景

业务已确认成功 (如拿到服务端 resourceId),后续的内存赋值/DB 持久化 必须完整执行,不能被取消中断;
仅成功路径执行,失败/取消不进入。

Android 实战代码

kotlin 复制代码
import kotlinx.coroutines.*
import androidx.lifecycle.lifecycleScope

// Android 页面中(Activity/Fragment)
lifecycleScope.launch(Dispatchers.IO) {
    // ========== 可取消区域 ==========
    // 耗时上传操作,cancel() 会立即断开 OkHttp
    val resourceId = uploadFileToServer()

    // ========== 仅业务成功进入:原子提交区 ==========
    if (resourceId != INVALID_ID) {
        // 屏蔽取消,保证落盘逻辑完整执行
        withContext(NonCancellable) {
            // 1. 内存状态赋值
            currentItem.remoteId = resourceId
            currentItem.status = UploadStatus.SUCCESS
            // 2. 本地持久化(DB/SP)
            saveUploadResultToDb(currentItem)
            // 3. 通知 UI 更新
            withContext(Dispatchers.Main) {
                updateUploadUi(currentItem)
            }
        }
    }
}

用法3:最佳实践 → 两种用法共存(Android 标准工程代码)

成功分支做原子提交,finally 做资源清理,互不冲突、各司其职。

kotlin 复制代码
lifecycleScope.launch(Dispatchers.IO) {
    var uploadConnection: UploadConnection? = null
    try {
        // 1. 可取消:建立连接 + 上传文件
        uploadConnection = createUploadConnection()
        val resourceId = uploadConnection.executeUpload()

        // 2. 用法2:业务成功 → 原子提交(你的项目核心逻辑)
        if (resourceId != INVALID_ID) {
            withContext(NonCancellable) {
                currentItem.remoteId = resourceId
                saveUploadResultToDb(currentItem)
            }
        }
    } finally {
        // 3. 用法1:官方规范 → 无论如何都释放资源
        withContext(NonCancellable) {
            uploadConnection?.close() // 安全关闭连接
        }
    }
}

配套:外部取消的标准写法(Android 环境)

kotlin 复制代码
// 取消上传任务(删除逻辑)
private fun cancelUploadJob(job: Job) {
    lifecycleScope.launch {
        job.cancel() // 立即中断上传(断开 OkHttp)
        job.join()   // 等待:原子提交块 + finally 清理 全部执行完毕
        // 执行后续删除逻辑
        deleteRemoteResource(currentItem.remoteId)
    }
}

总结

  1. 无唯一写法NonCancellable 不是必须写在 finally 中,官方从未规定唯一用法
  2. 两种标准场景
    • finally + NonCancellable官方规范 ,用于协程全生命周期的资源清理
    • 成功分支 + NonCancellable工程实战 ,用于业务结果的原子提交 (你的项目用法)
      取消后必须用 join() 等待不可取消代码执行完毕
相关推荐
程序员二叉4 分钟前
【Java】 异常高频面试题精讲 | 易错点+对比总结
java·开发语言·面试
慕木沐37 分钟前
Google ADK Java 1.0版本 核心机制与实战 Demo
java·开发语言·python
Roann_seo%43 分钟前
C++文件操作完全指南:从文本读写到二进制文件处理
开发语言·c++
JohnnyDeng941 小时前
【Android】Hilt 依赖注入:原理与最佳实践
android·kotlin·mvvm·hilt
huangdong_2 小时前
淘宝商品SKU图自动分类技术深度解析:从DOM解析到智能归档
开发语言·javascript·ecmascript
阿正的梦工坊2 小时前
【Rust】12-借用检查器与非词法生命周期
开发语言·后端·rust
qq_2518364572 小时前
基于java Web网络订餐系统设计与实现 源码文档
java·开发语言·前端
秋92 小时前
3年经验Python后端转AI Engineer:3个月实战转型计划(2026版)
开发语言·人工智能·python
凡人叶枫2 小时前
Effective C++ 条款17:以独立语句将 newed 对象置入智能指针
java·linux·开发语言·c++·算法
飞天狗1113 小时前
零基础JavaWeb入门——第2课:让网页“活”起来 —— JSP是什么?
java·开发语言·前端·后端·web