java程序员的kotlin课(N+1):coroutines 取消和超时

本文大部分翻译至:https://kotlinlang.org/docs/reference/coroutines/cancellation-and-timeouts.html
做了轻微优化
为什么翻译
我知道有一般中文版的文档,之所以还进行翻译有两个原因:

  • 看了好几遍,总是记不住,翻译一下,加深一下印象
  • 翻译的版本和原版的英文版,笔者认为讲的不够简单

为什么是N+1
kotlin系列文章照理说应该有一系列的文章,协程绝对应该是排在靠后的位置的,但是因为笔者最近一直在看这块的东西,而一些基础类的kotlin的文章反而没有写,所以协程系列文章以N开始,这是第二篇,所以是N+1

取消协程执行

在长时间执行的应用中,你也许需要对后台运行的协程有合适粒度的控制。举例来讲,比如用户已经关闭了某一个页面,那么后台运行的用来加载页面数据的协程就没有必要继续运行了并且应该被取消。lanch函数会返回一个job对象,这个job对象可以用来取消运行中的协程:

val job = launch {
    repeat(1000) { i ->
        println("job: I'm sleeping $i ...")
        delay(500L)
    }
}
delay(1300L) // delay a bit
println("main: I'm tired of waiting!")
job.cancel() // cancels the job
job.join() // waits for job's completion 
println("main: Now I can quit.")

这会产生如下的输出内容:

job: I'm sleeping 0 ...
job: I'm sleeping 1 ...
job: I'm sleeping 2 ...
main: I'm tired of waiting!
main: Now I can quit.

一但主函数执行了job.cancel,我们再也看不到协程有输出,因为它被取消了。job对象还有另外一个拓展函数cancelAndJoin,这个函数会融合cancel和join两个操作。

取消是需要配合的

协程的取消是需要内外一起配合的(此处有点拗口,后面不按照原文进行翻译了),啥意思呢?就是协程的取消,需要协程执行体内部配合外部的取消信号的;如果熟悉java的线程取消,大家可能会知道,如果是阻塞的方法,被取消会抛出·InterruptedException·,而非阻塞方法如果需要取消,必须在关键节点进行检查,检查线程当前是否被中断。协程也是一样的,如果一个协程内部没有suspend的代码,又没有在关键点设立检查点,协程是无法被取消的。比如下面这段代码,一个while循环没有设置任何协程取消检查点,所以在mian函数里调用了cancelAndJoin之后,协程内部依然不会停止执行。

val startTime = System.currentTimeMillis()
val job = launch(Dispatchers.Default) {
    var nextPrintTime = startTime
    var i = 0
    while (i < 5) { // computation loop, just wastes CPU
        // print a message twice a second
        if (System.currentTimeMillis() >= nextPrintTime) {
            println("job: I'm sleeping ${i++} ...")
            nextPrintTime += 500L
        }
    }
}
delay(1300L) // delay a bit
println("main: I'm tired of waiting!")
job.cancelAndJoin() // cancels the job and waits for its completion
println("main: Now I can quit.")

关于线程的取消,可以参考一下笔者早些年写的一篇java 线程池与通过Future终止线程实例

让协程可以被取消

让协程可以被取消,有两种思路:

  • 因为suspending的函数是可以被取消的,所以定期的调用一下suspending的方法,用来检查当前协程是否被取消。有一个函数yield是用来做这个的一个好选择
  • 另一个是显式的检查被取消的状态
    现在让我们来看下后一种方案:
val startTime = System.currentTimeMillis()
val job = launch(Dispatchers.Default) {
    var nextPrintTime = startTime
    var i = 0
    while (isActive) { // cancellable computation loop
        // print a message twice a second
        if (System.currentTimeMillis() >= nextPrintTime) {
            println("job: I'm sleeping ${i++} ...")
            nextPrintTime += 500L
        }
    }
}
delay(1300L) // delay a bit
println("main: I'm tired of waiting!")
job.cancelAndJoin() // cancels the job and waits for its completion
println("main: Now I can quit.")

现在如你所见,这个循环是可以被取消了。isActive是一个在CoroutineScope内部可用的拓展属性。

通过finally做取消后的善后工作

关闭suspending的函数,会抛出CancellationException异常,这个异常可以按照常规做法来进行处理,比如try {...} finally {...}

val job = launch {
    try {
        repeat(1000) { i ->
            println("job: I'm sleeping $i ...")
            delay(500L)
        }
    } finally {
        println("job: I'm running finally")
    }
}
delay(1300L) // delay a bit
println("main: I'm tired of waiting!")
job.cancelAndJoin() // cancels the job and waits for its completion
println("main: Now I can quit.")

join和cancelAndJoin函数会等待finally动作执行完毕,所以上面的例子,会输出如下:

job: I'm sleeping 0 ...
job: I'm sleeping 1 ...
job: I'm sleeping 2 ...
main: I'm tired of waiting!
job: I'm running finally
main: Now I can quit.

在finally里执行non-cancellable的代码

上面的代码演示了我们可以在finally的代码块中执行资源回收的操作,但是如果finally的代码块中执行suspend的代码会怎样?

val job = launch(Dispatchers.Default) {
    try {
      while (isActive){
        println("do sth.")
        delay(1000)
      }
    } finally {
      println("do in the finally")
      delay(10000)
      println("finally done.")
    }
  }

  delay(1300L) // delay a bit
  println("main: I'm tired of waiting!")
  job.cancelAndJoin() // cancels the job and waits for its completion
  println("main: Now I can quit.")

输出如下:

do sth.
do sth.
main: I'm tired of waiting!
do in the finally
main: Now I can quit.

finally done并没有被打印。
原因:我们的代码里在main中取消了协程的执行,协程内部的finally里又执行了delay方法,这个方法会使协程进入suspending状态,而协程被取消时suspending的执行函数会被取消。
但是因为finally里执行的都是需要做扫尾工作的动作,如果被取消,可能会造成资源泄漏问题,解决方案是用过withContext(NonCancellable){...}来包装扫尾工作的代码, 读者可以自己试一下。

超时

withTimeout(1300L) {
    repeat(1000) { i ->
        println("I'm sleeping $i ...")
        delay(500L)
    }
}

代码输出如下:

I'm sleeping 0 ...
I'm sleeping 1 ...
I'm sleeping 2 ...
Exception in thread "main" kotlinx.coroutines.TimeoutCancellationException: Timed out waiting for 1300 ms

代码打印了一个异常的堆栈信息,抛出了一个TimeoutCancellationException,它继承至CancellationException。我们之前在取消一个协程的时候,从来没有看到过console中有打印过这个异常信息,因为CancellationException被认为是协程内外用来做配合的常规异常。当然如果一个协程不是可取消的,那么timeout对它也是无可奈何的,比如下面这段代码:

withTimeout(1000) {
    launch {
      while (true){
        println(1)
      }
    }
  }
  println("done")

在看下面这段代码

val result = withTimeoutOrNull(1300L) {
    repeat(1000) { i ->
        println("I'm sleeping $i ...")
        delay(500L)
    }
    "Done" // will get cancelled before it produces this result
}
println("Result is $result")

这段代码因为使用了withTimeoutOrNull, 当超时时并不会抛出异常,而是会返回空。读者可以自己试一下。

系列文章快速导航:
java程序员的kotlin课(一):环境搭建
java程序员的kotlin课(N):coroutines基础
java程序员的kotlin课(N+1):coroutines 取消和超时
java程序员的kotlin课(N+2):suspending函数执行编排

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,378评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,356评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,702评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,259评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,263评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,036评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,349评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,979评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,469评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,938评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,059评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,703评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,257评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,262评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,501评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,792评论 2 345

推荐阅读更多精彩内容