你不知道的CoroutineContext:协程上下文大揭秘!

前言

协程(Coroutine)是一种并发编程技术,它允许我们在一个线程中执行多个任务,而不需要创建多个线程。协程与线程的区别在于,线程是操作系统的概念,而协程是编程语言的概念。协程可以暂停和恢复执行,而线程只能被终止。

在 Android 中,协程由 Kotlin 语言支持。Kotlin 协程库提供了丰富的 API,可以帮助我们轻松地编写并发代码。其中,CoroutineContext是一个非常重要的概念,它定义了协程的执行环境。

在本篇文章中,我们将从以下几个方面来介绍CoroutineContext的工作原理:

  • CoroutineContext的概念
  • CoroutineContext的组成
  • CoroutineContext的继承
  • CoroutineContext的注意事项

CoroutineContext的概念

CoroutineContext是一个容器,它包含了协程的所有上下文信息。这些上下文信息包括:

  • 协程的状态:协程的状态表示协程的生命周期。协程可以处于 ActiveCompletedCanceled 等状态。
  • 协程的调度策略:协程的调度策略决定了协程在哪里执行。协程可以执行在主线程、后台线程、或其他协程池中。
  • 协程的标签:协程的标签用于标识协程。
  • 协程的拦截器:协程的拦截器用于拦截协程的执行流程。
  • 协程的异常捕获:用于处理协程内部发生的未捕获异常。

CoroutineContext可以通过 coroutineContext获取。

fun main() = runBlocking {
    val context = coroutineContext

    println(context)
}

输出:

[CoroutineId(2), "coroutine#2":BlockingCoroutine{Active}@769c9116, BlockingEventLoop@6aceb1a5]

CoroutineContext的组成

CoroutineContext由多个组件组成,这些组件可以通过 context.get<T>() 函数来获取。

public operator fun <E : Element> get(key: Key<E>): E?

由于重新定义了get操作符,所以可以直接使用context[key]来获取对应的上下文组件元素。

  • Dispatcher:协程的调度策略。
fun main() = runBlocking {
    val context = coroutineContext + Dispatchers.Main

    val dispatcher = context[CoroutineDispatcher]

    println(dispatcher)
}

输出:

Dispatchers.Main[missing]

  • Job:协程的状态。Job 表示协程的生命周期。
fun main() = runBlocking {
    val context = coroutineContext + SupervisorJob()

    val job = context[Job]

    println(job)
}

输出:

SupervisorJobImpl{Active}@50675690

  • 获取协程的状态:协程的状态表示协程的生命周期。协程可以处于 ActiveCompletedCanceled 等状态。
fun main() = runBlocking {
    val context = coroutineContext + SupervisorJob()

    // 获取协程的状态
    val job = context[Job]

    // 判断协程是否处于 Active 状态
    if (job?.isActive == true) {
        println("协程处于 Active 状态")
    }
}

输出:

协程处于 Active 状态

  • CoroutineName:协程的标签。CoroutineName 用于标识协程。
fun main() = runBlocking {
    val context = coroutineContext + CoroutineName("张三")

    val coroutineName = context[CoroutineName]

    println(coroutineName)
}

输出:

CoroutineName(张三)

  • 添加拦截器:拦截器可以拦截协程的执行流程,例如:
  1. 在协程开始执行之前进行一些初始化操作。
  2. 在协程执行期间进行一些监控操作。
  3. 在协程执行完成之后进行一些清理操作。
class MyContinuationInterceptor : ContinuationInterceptor {

    override fun interceptContinuation(continuation: Continuation<Unit>): Continuation<Unit> {
        // 在协程开始执行之前进行一些初始化操作
        println("MyContinuationInterceptor: 协程开始执行之前")

        // 返回原始的 continuation
        return continuation
    }

    override fun key(): CoroutineContext.Key<ContinuationInterceptor> = ContinuationInterceptor.Key
}

fun main() {
    // 启动一个协程
    launch(Dispatchers.IO + MyContinuationInterceptor()) {
        // 执行一些耗时操作
        delay(1000)
    }
}

在这个示例中,协程在开始执行之前会打印一条消息:

MyContinuationInterceptor: 协程开始执行之前

  • CoroutineExceptionHandler:处理协程内部发生的未捕获异常
import kotlinx.coroutines.*

fun main() {
    // 创建CoroutineExceptionHandler
    val exceptionHandler = CoroutineExceptionHandler { _, exception ->
        println("Caught an exception: $exception")
    }

    // 启动一个协程,并指定CoroutineExceptionHandler
    runBlocking {
    	val context = coroutineContext + exceptionHandler
        val job = GlobalScope.launch(context) {
            // 模拟一个可能抛出异常的操作
            println("Coroutine is doing some work")
            delay(1000)
            throw CustomException("Something went wrong!")
        }

        // 等待协程执行结束
        job.join()
    }
}

// 自定义异常类
class CustomException(message: String) : Exception(message)


在这个示例中,为原有的coroutineContext增加了捕获异常的exceptionHandler,以至于协程内容抛出异常时,会被CoroutineExceptionHandler所捕获。

使用CoroutineExceptionHandler的好处在于,你可以集中处理协程内部的所有异常,而不必在每个协程体中都使用try-catch块来捕获异常。

  • EmptyCoroutineContext:一个空的 CoroutineContext。

CoroutineContext的继承

CoroutineContext支持继承。子CoroutineContext可以继承父CoroutineContext的所有组件。

fun main() = runBlocking {
    val parentContext = coroutineContext + Dispatchers.Main + SupervisorJob() + CoroutineName("张三")
    val childContext = parentContext + Dispatchers.IO

    println(childContext)
}

输出:

[CoroutineId(2), SupervisorJobImpl{Active}@1b40d5f0, CoroutineName(张三), Dispatchers.IO]

在这个例子中,parentContext 包含 Dispatchers.MainJob()CoroutineName("张三")childContext 继承了 parentContext 的所有组件,并添加了 Dispatchers.IO,由于与Dispatchers.Main同为调度器,所以最终保留的是最后的Dispatchers.IO

CoroutineContext的注意事项

在使用CoroutineContext时,需要注意以下几点:

  • 合理选择调度器:根据任务的性质选择合适的调度器,避免在IO密集型任务中使用CPU密集型的调度器,以及反之。
  • 细致管理CoroutineContext:合理管理CoroutineContext的元素,不要过度添加不必要的元素,以免引起不必要的性能开销。
  • 异常处理:及时处理协程中的异常,可以通过在CoroutineContext中添加CoroutineExceptionHandler元素来实现。

总结

总而言之,CoroutineContext是协程的一个重要概念。充分理解CoroutineContext的工作原理和使用方法,这样才能更好地利用CoroutineContext来控制协程的执行。

最后

如果想要成为架构师或想突破20~30K薪资范畴,那就不要局限在编码,业务,要会选型、扩展,提升编程思维。此外,良好的职业规划也很重要,学习的习惯很重要,但是最重要的还是要能持之以恒,任何不能坚持落实的计划都是空谈。

如果你没有方向,这里给大家分享一套由阿里高级架构师编写的《Android八大模块进阶笔记》,帮大家将杂乱、零散、碎片化的知识进行体系化的整理,让大家系统而高效地掌握Android开发的各个知识点。
img
相对于我们平时看的碎片化内容,这份笔记的知识点更系统化,更容易理解和记忆,是严格按照知识体系编排的。

欢迎大家一键三连支持,若需要文中资料,直接扫描文末CSDN官方认证微信卡片免费领取↓↓↓(文末还有ChatGPT机器人小福利哦,大家千万不要错过)

PS:群里还设有ChatGPT机器人,可以解答大家在工作上或者是技术上的问题

图片

  • 17
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值