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() 等待不可取消代码执行完毕
相关推荐
CYY9511 小时前
Compose 入门篇
android·kotlin
杉氧15 小时前
Compose 时代的 MVI 架构:如何用单向数据流驱动复杂 UI?
android·架构·android jetpack
杉氧15 小时前
Modifier 的艺术:为什么链式调用的顺序决定了UI 的生命周期?
android·架构·android jetpack
李斯维16 小时前
腾讯 XLog 日志框架 Android 端接入
android·android studio·android jetpack
黄林晴16 小时前
Kotlin Toolchain 0.11 发布:Amper 正式更名,统一 kotlin 命令
android·kotlin
雨白17 小时前
C语言基础快速入门与指针初探
android
Exploring19 小时前
避坑指南:升级 AGP 8.0+ 导致第三方 SDK 编译崩溃的完美解决方案
android
石山岭2 天前
自己动手写了一个 Android 虚拟定位 App:GPSSimulate 技术实
android·前端
杉氧2 天前
副作用 (Side Effects) 全攻略:如何像大师一样掌控 Composable 的生命周期?
android·架构·android jetpack
唐青枫2 天前
别再把 inline 当性能开关:Kotlin 内联、noinline、crossinline 与 reified 实战详解
kotlin