美文网首页
4.协程的异常处理(2)

4.协程的异常处理(2)

作者: 学吉他的袁先生 | 来源:发表于2020-08-08 10:01 被阅读0次

异常的传播
异常传播是指异常在父子协程中的传播,什么是父子协程,在当前协程中又启动一个协程,那么这个新启动的协程就是当前协程的子协程。异常的传播涉及到协程作用域的概念
一协程作用域

1.协程作用域本身是一个接口

public interface CoroutineScope {
    //此作用域的上下文
    //上下文由作用域封装,用于实现协程生成器,这些生成器是作用域上的扩展
    //此属性除了为了高级用途而访问[Job]实例外,不推荐访问
    //通常应该包含一个Job的实例来执行结构化并发
    public val coroutineContext: CoroutineContext
}

子类如下


clipboard.png

看一个例子,来加深理解作用域的父子关系

GlobalScope.launch {
    println("GlobalScope ${this}")
    launch {
        println("A ${this}")
        launch {
            println("A1 ${this}")
        }
    }

    launch {
        println("B ${this}")
    }
}.join()

打印

GlobalScope StandaloneCoroutine{Active}@4957b7c3
A StandaloneCoroutine{Active}@507cdd7a
B StandaloneCoroutine{Active}@64a45f82
A1 StandaloneCoroutine{Active}@544353ad

关系如下图所示


4232467852.png

作用域启动新协程也是一个新的作用域,它们的关系可以并列,也可以包含,组成了一个作用域的树形结构默认情况下,每个协程都要等待它的子协程全部完成后,才能结束自己。这种形式,就被称为结构化的并发

2.我们之前启动协程一直用的是GlobalScope,GlobalScope是一个顶级的协程作用域,此外还有coroutineScope{...}以及 supervisorScope{...}等,这里我们重点讲一下跟异常传播有关系的coroutineScope{...}跟supervisorScope{...}
coroutineScope源码

//创建协程作用域并在此范围调用指定的代码块
//该作用域继承了外部作用域的协程上下文但是重写了里面的Job
//这个函数是为分解并行工作而设计的
//当这个作用域内的任何子协同程序失败时,所有其余的子协程都被取消
public suspend fun <R> coroutineScope(block: suspend CoroutineScope.() -> R): R =
    suspendCoroutineUninterceptedOrReturn { uCont ->
        val coroutine = ScopeCoroutine(uCont.context, uCont)
        coroutine.startUndispatchedOrReturn(coroutine, block)
    }

来看一下异常传播对于coroutineScope{...}中是怎样的

coroutineScope {
    launch {
        log(1)
        launch {
            log(2)
            throw ArithmeticException()
        }
    }
    launch {
        delay(200)
        log(3)
    }
}

打印如下

18:19:32:955 [DefaultDispatcher-worker-1] 1
18:19:32:955 [DefaultDispatcher-worker-2] 2
Exception in thread "main" java.lang.ArithmeticException

coroutineScope 当中协程异常会触发父协程的取消,进而将整个协程作用域取消掉,如果对 coroutineScope 整体进行捕获,也可以捕获到该异常

supervisorScope源码

//创建协程作用域并在此作用域调用指定的代码块
//该作用域继承了外部作用域的协程上下文但是重写了里面的Job
//子协程的失败不会导致此作用域失败,也不会影响其他子协程
public suspend fun <R>  supervisorScope(block: suspend CoroutineScope.() -> R): R =
    suspendCoroutineUninterceptedOrReturn { uCont ->
        val coroutine = SupervisorCoroutine(uCont.context, uCont)
        coroutine.startUndispatchedOrReturn(coroutine, block)
}

来看一下异常传播在supervisorScope中是怎样的

supervisorScope{
    launch {
        log(1)
        throw ArithmeticException()
    }
    launch {
        delay(200)
        log(2)
    }
}

打印如下

18:33:15:486 [DefaultDispatcher-worker-1] 1
18:33:15:523 [DefaultDispatcher-worker-1] 2
Exception in thread "DefaultDispatcher-worker-1" java.lang.ArithmeticException

如何捕获异常如下

val exceptionHandler = CoroutineExceptionHandler { coroutineContext, throwable ->
    log(throwable)
}
supervisorScope{
    launch (exceptionHandler){
        log(1)
        throw ArithmeticException()
    }
    launch {
        delay(200)
        log(2)
    }
}

相关文章

  • 4.协程的异常处理(2)

    异常的传播异常传播是指异常在父子协程中的传播,什么是父子协程,在当前协程中又启动一个协程,那么这个新启动的协程就是...

  • 4.协程的异常处理(1)

    1.讲解异常之前先写这样一个例子需求异步操作获取用户,主界面展示用户信息,怎么样用协程的方式去写 思考如果获取us...

  • Android中对Kotlin Coroutines(协程)的理

    (接第一部分) 异常处理1.协程的异常传递协程的异常传播也是遵循了协程上下文的机制,除了取消异常(Cancella...

  • Kotlin协程

    协程基础概念[https://zhuanlan.zhihu.com/p/427092689]协程异常处理[http...

  • Android版kotlin协程入门(三):kotlin协程的异

    kotlin协程的异常处理 在上一篇《Android kotlin协程入门(二):kotlin协程的关键知识点初步...

  • Python协程

    目录:一、基于生成器的协程二、协程状态三、协程预激装饰器四、终止协程和异常处理五、协程返回值六、yield fro...

  • 协程中的取消和异常 (核心概念介绍)

    当我们需要避免多余的协程处理任务减少内存浪费,节约电量,协程的取消操作尤为重要。而协程的异常处理对于用户体验来说也...

  • 协程的异常处理

    CoroutineExceptionHandler 被用来将通用的 catch代码块在协程中自定义日志记录或异常处...

  • 协程的异常处理

    什么情况下 try-catch 会失效? 在 try-catch 块中创建了一个子协程,调用了一个百分百会失败的接...

  • 破解 Kotlin 协程(4) - 异常处理篇

    关键词:Kotlin 协程 异常处理 异步代码的异常处理通常都比较让人头疼,而协程则再一次展现了它的威力。 1. ...

网友评论

      本文标题:4.协程的异常处理(2)

      本文链接:https://www.haomeiwen.com/subject/azgcdktx.html