Kotlin编程第一课--(协程篇)23 异常:try-catch 坑!
这节课我们来学习 Kotlin 协程的异常处理。
其实到这里,我们就已经学完所有 Kotlin 协程的语法知识了。但在真正把 Kotlin 协程应用到生产环境之前,我们还需要掌握一个重要知识点,那就是异常处理。
比起 Kotlin 协程的语法知识点,协程的异常处理,其实更难掌握。在前面的课程中,我们已经了解到:协程就是互相协作的程序,协程是结构化的。正因为 Kotlin 协程有这两个特点,这就导致它的异常处理机制与我们普通的程序完全不一样。
换句话说:如果把 Java 里的那一套异常处理机制,照搬到 Kotlin 协程里来,你一定会四处碰壁。因为在普通的程序当中,你使用 try-catch 就能解决大部分的异常处理问题,但是在协程当中,根据不同的协程特性,它的异常处理策略是随之变化的。
我自己在工作中就踩过很多这方面的坑,遇到过各种匪夷所思的问题:协程无法取消、try-catch 不起作用导致线上崩溃率突然大增、软件功能错乱却追踪不到任何异常信息,等等。说实话,Kotlin 协程的普及率之所以不高,很大一部分原因也是因为它的异常处理机制太复杂了,稍有不慎就可能会掉坑里去。
那么今天这节课,我们就会来分析几个常见的协程代码模式,通过解决这些异常,我们可以总结出协程异常处理的 6 大准则。掌握了这些准则之后,你在以后遇到异常问题时,就能有所准备,也知道该怎么处理了。
为什么 cancel() 不起作用?
在 Kotlin 协程当中,我们通常把异常分为两大类,一类是取消异常(CancellationException),另一类是其他异常。之所以要这么分类,是因为在 Kotlin 协程当中,这两种异常的处理方式是不一样的。或者说,在 Kotlin 协程所有的异常当中,我们需要把 CancellationException 单独拎出来,特殊对待。
要知道,当协程任务被取消的时候,它的内部是会产生一个 CancellationException 的。而协程的结构化并发,最大的优势就在于:如果我们取消了父协程,子协程也会跟着被取消。但是我们也知道,很多初学者都会遇到一个问题,那就是协程无法被取消。
这里,主要涉及了三个场景,我们一个个来分析下。
场景 1:cancel() 不被响应
|
在上面的代码中,我们启动了一个协程,在这个协程的内部,我们一直对 i 进行自增。过了 2000 毫秒以后,我们调用了 job.cancel()。但通过运行的结果,我们可以看到协程并不会被取消。这是为什么呢?
其实前面课程里我们就讲过,协程是互相协作的程序。因此,对于协程任务的取消,也是需要互相协作的。协程外部取消,协程内部需要做出响应才行。具体来说,我们可以在协程体中加入状态判断:
|
在这段代码里,我们把 while 循环的条件改成了 while (isActive),这就意味着,只有协程处于活跃状态的时候,才会继续执行循环体内部的代码。
这里,我们就可以进一步分析代码段 1 无法取消的原因了:当我们调用 job.cancel() 以后,协程任务已经不是活跃状态了,但代码并没有把 isActive 作为循环条件,因此协程无法真正取消。
所以到这里,我们就可以总结出协程异常处理的第一准则了:协程的取消需要内部的配合。
场景 2:结构被破坏
我们都知道,协程是结构化的,当我们取消父协程的时候,子协程也会跟着被取消。比如,我们在第 16 讲当中,就看到过这张图:
但在某些情况下,我们嵌套创建的子协程并不会跟随父协程一起取消,比如下面这个案例:
|
以上代码中,我们创建了一个 fixedDispatcher,它是由两个线程的线程池实现的。接着,我们通过 launch 创建了三个协程,其中 parentJob 是父协程,随后我们等待 2000 毫秒,然后取消父协程。
不过,通过程序的运行结果,我们发现,虽然“子协程 1”当中使用了 while(isActive) 作为判断条件,它也仍然无法被取消。其实,这里的主要原因还是在注释 1 处,我们在创建子协程的时候,**使用了 launch(Job()){}**。而这种创建方式,就打破了原有的协程结构。
为了方便你理解,我画了一张图,描述它们之间的父子关系。
根据这张图,可以看到“子协程 1”已经不是 parentJob 的子协程了,而对应的,它的父 Job 是我们在 launch 当中传入的 Job() 对象。所以,在这种情况下,当我们调用 parentJob.cancel() 的时候,自然也就无法取消“子协程 1”了。
其实这个时候,如果我们稍微改动一下上面的代码,不传入 Job(),程序就可以正常运行了。
|
在上面的代码中,parentJob 与它内部的子协程 1、子协程 2 之间是父子关系,因此它们两个都是会响应协程取消的事件的。这时候,它们之间的关系就变成了下图这样:
那么到这里,我们其实就可以总结出第二条准则了:不要轻易打破协程的父子结构!
场景 3:未正确处理 CancellationException
其实,对于 Kotlin 提供的挂起函数,它们是可以自动响应协程的取消的,比如说,当我们把 Thread.sleep(500) 改为 delay(500) 以后,我们就不需要在 while 循环当中判断 isActive 了。
|
实际上,对于 delay() 函数来说,它可以自动检测当前的协程是否已经被取消,如果已经被取消的话,它会抛出一个 CancellationException,从而终止当前的协程。
为了证明这一点,我们可以在以上代码的基础上,增加一个 try-catch。
|
请看注释 1,在用 try-catch 包裹了 delay() 以后,我们就可以在输出结果中,看到“Catch CancellationException”,这就说明 delay() 确实可以自动响应协程的取消,并且产生 CancellationException 异常。
请看注释 1,在用 try-catch 包裹了 delay() 以后,我们就可以在输出结果中,看到“Catch CancellationException”,这就说明 delay() 确实可以自动响应协程的取消,并且产生 CancellationException 异常。
|
可见,在这段代码中,我们把“throw e”这行代码注释掉,重新运行之后,程序就永远无法终止了。这主要是因为,我们捕获了 CancellationException 以后没有重新抛出去,就导致子协程无法正常取消。所以到这里,
我们就可以总结出第三条准则了:捕获了 CancellationException 以后,要考虑是否应该重新抛出来。
题外话:很多开发者喜欢在代码里捕获 Exception 这个父类,比如这样:catch(e: Exception){},这也是很危险的。平时写 Demo 为了方便这样写没问题,但在生产环境则应该禁止。
好,到这里,我们就通过协程取消异常的三个场景,总结了三条准则,来应对 CancellationException 这个特殊的异常。
那么接下来,我们再来看看如何在协程当中处理普通的异常。
为什么 try-catch 不起作用?
如果你有 Java 经验,那你一定会习惯性地把 try-catch 当做是解决所有异常的手段。但是,在 Kotlin 协程当中,try-catch 并非万能的。有时候,即使你用 try-catch 包裹了可能抛异常的代码,软件仍然会崩溃。比如下面这个例子:
|
在这段代码中,我们使用 try-catch 包裹了 launch{},在协程体内部,我们制造了一个异常。不过从运行结果这里,我们可以看到,try-catch 并没有成功捕获异常,程序等待了 100 毫秒左右,最终还是崩溃了。
类似的,如果我们把代码段 8 当中的 launch 换成 async,结果也是差不多的:
|
其实,对于这种 try-catch 失效的问题,如果你还记得在第 14 讲当中,我提到的 launch、async 的代码运行顺序的问题,那你就一定可以理解其中的原因。这主要就是因为,当协程体当中的“1/0”执行的时候,我们的程序已经跳出 try-catch 的作用域了。
当然,要解决这两个问题也很容易。对于代码段 8 来说,我们可以挪动一下 try-catch 的位置,比如说这样:
|
OK,到这里,我们就可以总结出第四条准则了:不要用 try-catch 直接包裹 launch、async。
接下来,我们再看看 async 的另外一种手段,其实这种方式网上有些博客也介绍过,我们可以使用 try-catch 包裹“deferred.await()”。让我们来看看是否可行:
|
那么,根据以上程序的运行结果可以看到,这样做其实是行不通的。如果你看过一些其他博客,甚至还有种说法是:await() 如果不调用的话,async 当中的异常甚至不会发生。我们再来试试看:
|
可见,async 当中产生异常,即使我们不调用 await() 同样是会导致程序崩溃的。那么,为什么会发生这样的情况?是不是我们忽略了什么?
SupervisorJob
实际上,如果我们要使用 try-catch 包裹“deferred.await()”的话,还需要配合 SupervisorJob 一起使用。也就是说,借助 SupervisorJob 来改造代码段 13 的话,我们就可以实现“不调用 await() 就不会产生异常而崩溃”。
|
可以看到,当我们使用 SupervisorJob 创建一个 scope 以后,用 scope.async{}启动协程后,只要不调用“deferred.await()”,程序就不会因为异常而崩溃。
所以同样的,我们也能用类似的办法来改造代码段 12 当中的逻辑:
|
在上面的代码中,我们仍然使用“scope.async {}”创建了协程,同时也用 try-catch 包裹“deferred.await()”,这样一来,这个异常就成功地被我们捕获了。
那么,SupervisorJob 到底是何方神圣?让我们来看看它的源码定义:
|
根据以上代码,我们可以看到,SupervisorJob() 其实不是构造函数,它只是一个普通的顶层函数。而这个方法返回的对象,是 Job 的子类。
SupervisorJob 与 Job 最大的区别就在于,当它的子 Job 发生异常的时候,其他的子 Job 不会受到牵连。我这么说你可能会有点懵,下面我做了一个动图,来演示普通 Job 与 SupervisorJob 之间的差异。
这个是普通 Job,对于子 Job 出现异常时的应对策略。可以看到,由于 parentJob 是一个普通的 Job 对象,当 job1 发生异常之后,它会导致 parentJob 取消,进而导致 job2、job3 也受到牵连。
而这时候,如果我们把 parentJob 改为 SupervisorJob,job1 发生异常的的话,就不会影响到其他的 Job 了。
所以到这里,我们就可以总结出第五条准则了:灵活使用 SupervisorJob,控制异常传播的范围。
提示:并非所有情况下,我们都应该使用 SupervisorJob,有时候 Job 会更合适,这要结合实际场景分析。
好,到目前为止,我们就已经了解了 try-catch 和 SupervisorJob 这两种处理异常的手段。但是,由于协程是结构化的,当我们的协程任务出现复杂的层级时,这两种手段其实都无法很好的应对。所以这个时候,我们就需要 CoroutineExceptionHandler 出场了。
CoroutineExceptionHandler
对于 CoroutineExceptionHandler,我们其实在第 17 讲里也简单地提到过。它是 CoroutineContext 的元素之一,我们在创建协程的时候,可以指定对应的 CoroutineExceptionHandler。
那么 CoroutineExceptionHandler 究竟适用于什么样的场景呢?让我们来看一个例子:
|
在上面的代码中,我模拟了一个复杂的协程嵌套场景。对于这样的情况,我们其实很难一个个在每个协程体里面去写 try-catch。所以这时候,为了捕获到异常,我们就可以使用 CoroutineExceptionHandler 了。
|
以上代码中,我们定义了一个 CoroutineExceptionHandler,然后把它传入了 scope 当中,这样一来,我们就可以捕获其中所有的异常了。
看到这里,你也许松了一口气:终于有了一个简单处理协程异常的方式了。不过,你也别高兴得太早,因为我曾经就踩过 CoroutineExceptionHandler 的一个坑,最终导致 App 功能大面积异常。
而出现这个问题的原因就是:CoroutineExceptionHandler 不起作用了!
为什么 CoroutineExceptionHandler 不起作用?
为了模拟我当时的业务场景,我把代码段 18 稍作改动。
|
请你留意上面的注释,我们把自定义的 myExceptionHandler,放到出现异常的 launch 那里传了进去。按理说,程序的执行结果是不会发生变化才对的。但实际上,myExceptionHandler 并不会起作用,我们的异常不会被它捕获。
如果你对比代码段 18 和代码段 19,你会发现,myExceptionHandler 直接定义在发生异常的位置反而不生效,而定义在最顶层却可以生效!你说它的作用域是不是很古怪?
其实,出现这种现象的原因,就是因为:CoroutineExceptionHandler 只在顶层的协程当中才会起作用。也就是说,当子协程当中出现异常以后,它们都会统一上报给顶层的父协程,然后顶层的父协程才会去调用 CoroutineExceptionHandler,来处理对应的异常。
那么到这里,我们就可以总结出第六条准则了:使用 CoroutineExceptionHandler 处理复杂结构的协程异常,它仅在顶层协程中起作用。
小结
至此,这节课的内容就接近尾声了,我们来做一个简单的总结。
在 Kotlin 协程当中,异常主要分为两大类,一类是协程取消异常(CancellationException),另一类是其他异常。为了处理这两大类问题,我们一共总结出了 6 大准则,这些我们都要牢记在心。
- 第一条准则:协程的取消需要内部的配合。
- 第二条准则:不要轻易打破协程的父子结构!这一点,其实不仅仅只是针对协程的取消异常,而是要贯穿于整个协程的使用过程中。我们知道,协程的优势在于结构化并发,它的许多特性都是建立在这个特性之上的,如果我们无意中打破了它的父子结构,就会导致协程无法按照预期执行。
- 第三条准则:捕获了 CancellationException 以后,要考虑是否应该重新抛出来。在协程体内部,协程是依赖于 CancellationException 来实现结构化取消的,有的时候我们出于某些目的需要捕获 CancellationException,但捕获完以后,我们还需要思考是否需要将其重新抛出来。
- 第四条准则:不要用 try-catch 直接包裹 launch、async。这一点是很多初学者会犯的错误,考虑到协程代码的执行顺序与普通程序不一样,我们直接使用 try-catch 包裹 launch、async,是不会有任何效果的。
- 第五条准则:灵活使用 SupervisorJob,控制异常传播的范围。SupervisorJob 是一种特殊的 Job,它可以控制异常的传播范围。普通的 Job,它会因为子协程当中的异常而取消自身,而 SupervisorJob 则不会受到子协程异常的影响。在很多业务场景下,我们都不希望子协程影响到父协程,所以 SupervisorJob 的应用范围也非常广。比如说 Android 当中的 viewModelScope,它就使用了 SupervisorJob,这样一来,我们的 App 就不会因为某个子协程的异常导致整个应用的功能出现紊乱。
- 第六条准则:使用 CoroutineExceptionHandler 处理复杂结构的协程异常,它仅在顶层协程中起作用。我们都知道,传统的 try-catch 在协程当中并不能解决所有问题,尤其是在协程嵌套层级较深的情况下。这时候,Kotlin 官方为我们提供了 CoroutineExceptionHandler 作为补充。有了它,我们可以轻松捕获整个作用域内的所有异常。
其实,这节课里我提到的这些案例,只是我平时工作中遇到的很小一部分。案例是讲不完的,在协程中处理异常,你将来肯定也会遇到千奇百怪的问题。但重要的是分析问题的思路,还有解决问题的手段。这节课我给你总结的 6 大准则呢,就是你将来遇到协程异常时,可以用的 6 种处理手段。
当我们遇到问题的时候,首先要分析是 CancellationException 导致的,还是其他异常导致的。接着我们就可以根据实际情况去思考,该用哪种处理手段了。
另外如果你足够细心的话,你会发现这节课总结出的 6 大准则,其实都跟协程的结构化并发有着密切联系。由于协程之间存在父子关系,因此它的异常处理也是遵循这一规律的。而协程的异常处理机制之所以这么复杂,也是因为它的结构化并发特性。
所以,除了这 6 大准则以外,我们还可以总结出一个核心理念:因为协程是“结构化的”,所以异常传播也是“结构化的”。
如果你能理解协程异常处理的核心理念,同时能够牢记前面的 6 大准则。我相信,将来不论你遇到什么样的古怪问题,你都可以分析出问题的根源,找到解决方案!
思考题
前面我们提到过,CoroutineExceptionHandler 可以一次性捕获整个作用域内所有协程的异常。那么,我们是不是可以抛弃 try-catch,只使用 CoroutineExceptionHandler 呢?为什么?