协程生命周期的最后一步---协程取消

任何对象都有生命周期,协程也不例外,其生命周期很简单启动->运行->结束。而每个生命周期的状态转换都是需要触发条件的,比如启动->运行,需要协程构建器launch{},运行期间需要续体,线程池等,最后结束的直接触发条件就是本文要探讨的内容

线程的取消

因为协程的取消和线程的取消在原理上是非常相近的,而线程又是读者比较容易接受的知识,所以在探讨协程的取消前,先来看线程是如何取消的。

如何取消线程
  • stopsuspend:在底层上存在严重的缺陷,应该避免使用此类方式停止线程
  • 中断interrupted方法:这是一种协作机制,简言之就是线程A设置一个线程B的中断的flag,B在耗时循环体内检查该flag,true则放弃该循环,则线程B会自行停止。提供如下实例代码,简单说明其协作原理:
public class MyThread extends Thread {
    @Override
    public void run() {
        super.run();
        try {
            for (int i = 0; i < 500000; i++) {
                // check中断标志位
                if (this.isInterrupted()) {
                    System.out.println("stop");
                    // 异常法,使线程自行停止
                    throw new InterruptedException();
                }
            }
        } catch (InterruptedException e) {
            System.out.println("run catch");
            // to do someThing
        }
    }

    public static void main(String[] args) {
        try {
            Thread.sleep(2000);
            MyThread thread = new MyThread();
            thread.start();
            // 停止线程,设置中断标志位
            thread.interrupt();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

附:这样我们就知道官方协程指南中提到"取消是协作的”是什么意思了,其实原理和这个类似。

协程的取消

和线程的取消一样,协程的取消也是协作机制,类似线程的interrupt()函数,协程可以调用cancel()函数取消,或者cancelAndJoin()。类似isInterrupted(), 协程可以check isActive的值去判断协程是否取消。

cancelcancelAndJoin()
  • cancel直接取消协程,即在耗时循环体内抛CancellationException异常,不会等待耗时循环体后面的代码执行完成, 如下例子可表明
 suspend fun test2() {
    coroutineScope {
        val job = launch(Dispatchers.Default) {
            log("1")
            val file = File("E:\\Opera_64.0.3417.54_Setup_x64.exe")
            val bufferedReader = BufferedReader(FileReader(file))
            // 耗时循环体:大多数IO耗时操作都会有一个循环的
            while (isActive) {
                bufferedReader.readLine() ?: break
            }
            log("2")
        }
        delay(100L) // 等待一段时间
        job.cancel()
        log("3")
    }
}
// output
Thread[DefaultDispatcher-worker-1,5,main]1
Thread[main,5,main]3
Thread[DefaultDispatcher-worker-1,5,main]2
Thread[main,5,main]end
  • cancelAndJoin()是会等待耗时循环体后面的代码执行完成的,这个一般不会花太多时间,因为最耗时的操作已经通过判断isActive跳过了。一般会使用这个函数去处理。如下代码的输出顺序可证明该结论:
suspend fun test2() {
    coroutineScope {
        val job = launch(Dispatchers.Default) {
            log("1")
            val file = File("E:\\Opera_64.0.3417.54_Setup_x64.exe")
            val bufferedReader = BufferedReader(FileReader(file))
            while (isActive) {
                bufferedReader.readLine() ?: break
            }
            log("2")
        }
        delay(100L) // 等待一段时间
        job.cancelAndJoin()
        log("3")
    }
}
// output
Thread[DefaultDispatcher-worker-1,5,main]1
Thread[DefaultDispatcher-worker-1,5,main]2
Thread[main,5,main]3
Thread[main,5,main]end

可能到这里有人会说了,那cancelcancelJoin这两个api有什么用呢?协作式的停止协程需要自己去写判断条件呀?我的回答是yes,你需要自己写,不过幸运的是,耗时操作一般是网络+文件读取,其实说白了就是IO,这两种场景都有现成的框架,比如网络有Retrofit等,这些经典的框架对cancel这样的操作是做了适配的。即使没有做适配,你也可以自己简单的写一个带取消协程。步骤如下:

  • 首先你需要理解suspendCancellableCoroutine:类似上篇讲的suspendCoroutineUninterceptedOrReturn其回调也会返回一个续体
suspend fun test4() = suspendCancellableCoroutine<String> { cont ->
    // 定义一个取消回调事件
    cont.invokeOnCancellation {
        // 在这里可以做一些耗时操作的cancel处理
    }
}
  • OkHttp为例,就可以在invokeOnCancellation回调中调用call.cancel去真正的取消该请求。
suspend fun test4() = suspendCancellableCoroutine<String> { cont ->
    val call = OkHttpClient().newCall(...)
    // 定义一个取消回调事件
    cont.invokeOnCancellation {
        // 取消请求
        call.cancel()
    }
}

这就完成了,我们在主函数的调用代码如下

val job = launch { //①
    log(1)
    val res = test4()
    log(res)
    log(2)
}
delay(10)
log(3)
job1.cancel()
log(4)

(附:Retrofit从2.6.0版,已经对其做了适配, 也是类似上面的原理。后面的协程实践篇我会详细介绍)

上面可能表述有点乱,所以现在总结一下cancelcancelJoin()到底做了什么

小结

取消协程主要有两个方法cancel()cancelJoin(),前者立即停止协程执行并执行后面代码,后者需要等待协程执行完成后在执行后面的代码。其中cancel主要做了两件事

  • 状态转移:即设置状态flag: isActive = false
  • 处理回调:通知各个观察者(订阅者)
    join主要就是干了一件事,即等待设置了isActive=false的协程执行完成。

思考与总结

很多对象的生命周期结束的问题会使程序设计和实现等过程变得复杂,比如在应用层你可能需要在结束时释放资源,在框架层你需要考虑内部的状态迁移和资源释放的问题。但是该步骤又是非常重要的,比如在android开发中就可能因为此造成内存泄漏等现象。所以在应用层设计的时候一定要考虑取消的情况,本文开篇通过引入线程的取消机制(协作机制)为后文描述协程的取消做铺垫,因为两者非常类似,

  • 线程的interrupt()等同于协程中的cancel()方法
  • 线程中判断是否中断的方法isInterrupted()等同于协程中的isActive

但是由于协作机制的复杂性(需要协作)导致在很多情况下需要自己在协程A中取消,在协程B中需要手动判断协程是否取消来跳过循环耗时函数。这就需要某些异步框架中对该机制做适配,比如RetrofitRxJava做了适配,当然retrofit也对协程做了适配。对于没有做适配的框架,本文也给出了一个的简单demo自己去做适配。

最后略显遗憾的是,没有对cancel进行源码级的分析,后面有机会补上。

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