Kotlin 协程 coroutineScope 和 supervisorScope 的区别

Kotlin 协程 coroutineScope 和 supervisorScope 的区别

  • coroutineScope 和 supervisorScope 都是用于创建协程作用域的函数,核心区别主要体现在异常传播机制和子协程的生命周期管理上
  • 两者都是挂起函数,父协程(即 coroutineScope 和 supervisorScope 所在的协程)会等待其作用域内的所有子协程(通过 launch 或 async 启动)完成后才返回,确保作用域内的所有操作执行完毕(通过挂起当前协程来等待子协程完成,不会阻塞线程)
  • 结构化并发:如果父协程被取消,所有子协程会被取消

coroutineScope

  • 协同作用域(coroutine 协同),使用普通 Job 管理子协程
  • 异常传播机制:子协程的异常会向上传播到父协程处理(父协程会被取消),导致整个作用域取消,其他子协程(包括未完成的)都会被取消
  • 应用场景:原子性操作,所有子协程要么全部成功,要么全部失败(比如并行下载一个文件、事务操作等)
kotlin 复制代码
//最终结果:作用域因为子协程 1 的异常而整体失败
suspend fun coroutineScopeTest() =
    coroutineScope {
        launch {
            println("子协程 1 开始")
            delay(100)
            throw RuntimeException("子协程 1 失败")
        }
        launch {
            println("子协程 2 开始")
            delay(200)
            println("子协程 2 尝试完成...")
        }
    }
//-------------- 打印 --------------
//子协程 1 开始
//子协程 2 开始
//Exception in thread "main" java.lang.RuntimeException: 子协程 1 失败

supervisorScope

  • 主从作用域(supervisor 主管),使用 SupervisorJob 管理子协程(通过 SupervisorJob 实现异常隔离,SupervisorJob 重写了 childCancelled 返回 false)
  • 异常传播机制:子协程的异常不会传播到父协程(父协程继续运行),不会取消整个作用域,仅取消自身,其他子协程会继续执行
  • 应用场景:子任务相互独立,允许部分子协程失败(比如并行下载多个文件,日志上传与数据保存并行等)
  • 子协程的异常需在子协程内部通过 try-catch 处理或者通过 CoroutineExceptionHandler 统一处理
  • viewModelScope 默认使用结构化并发,如果想要实现异常隔离,可以显式包裹一层 supervisorScope 实现
  • supervisorScope 只能作用一层
kotlin 复制代码
//最终结果:作用域等待所有子协程完成(子协程 2 成功,子协程 1 失败)
suspend fun supervisorScopeTest() =
    supervisorScope {
        launch {
            println("子协程 1 开始")
            delay(100)
            throw RuntimeException("子协程 1 失败")
        }
        launch {
            println("子协程 2 开始")
            delay(200)
            println("子协程 2 成功完成!")
        }
    }
//-------------- 打印 --------------
//子协程 1 开始
//子协程 2 开始
//Exception in thread "main" java.lang.RuntimeException: 子协程 1 失败
//子协程 2 成功完成!
kotlin 复制代码
//最终结果:supervisorScope 只能作用一层
suspend fun supervisorScopeTest2() =
    supervisorScope {
        launch {
            println("子协程 1 开始")
            launch {
                println("子子协程 A 开始")
                delay(100)
                throw RuntimeException("子子协程 A 失败")
            }
            launch {
                println("子子协程 B 开始")
                delay(200)
                println("子子协程 B 成功完成!")
            }
            delay(400)
            println("子协程 1 成功完成!")
        }
        launch {
            println("子协程 2 开始")
            delay(500)
            println("子协程 2 成功完成!")
        }
    }
//-------------- 打印 --------------
//子协程 1 开始
//子协程 2 开始
//子子协程 A 开始
//子子协程 B 开始
//Exception in thread "main" java.lang.RuntimeException: 子子协程 A 失败
//子协程 2 成功完成!
相关推荐
JMchen1231 天前
Android CameraX深度解析:从Camera1到CameraX的相机架构演进
android·java·数码相机·架构·kotlin·移动开发·android-studio
倔强的石头1061 天前
【Linux指南】进程控制系列(五)实战 —— 微型 Shell 命令行解释器实现
linux·运维·kotlin
Hz4532 天前
Android Jetpack核心组件协同实战:Navigation 3.X+Lifecycle+Flow+Hilt的架构革新
android·kotlin
JMchen1232 天前
Android音频编码原理与实践:从AAC到Opus,深入解析音频编码技术与移动端实现
android·经验分享·学习·kotlin·android studio·音视频·aac
JMchen1232 天前
Android音频处理全解析:从3A算法到空间音频,打造专业级音频体验
android·经验分享·算法·kotlin·android studio·音视频
瓦特what?2 天前
C++编程防坑指南(小说版)
android·c++·kotlin
一招定胜负2 天前
卷积神经网络提取人脸五个特征点
人工智能·cnn·kotlin
HeDongDong-2 天前
详解 Kotlin 的函数
开发语言·python·kotlin
zhangphil3 天前
Kotlin高阶函数及函数作为参数传递(2)
kotlin
Yang-Never3 天前
Open GL ES -> 应用前后台、Recent切换,SurfaceView纹理贴图闪烁问题分析解决
android·开发语言·kotlin·android studio·贴图