kotlin 协程实例分析(2)

我们要说明一下,这里的实例分析并不会去分析协程的源码,所以说这个是个使用层面的分析。因为我本来就是个使用者,不太关心它是怎么实现的, 我们此次要进一步了解协程的执行顺序,调度原理,还是先出题,说出下面实例的输出顺序,如能说出代表你对协程已经达到入门级,不能的话看第一个实例

package com.cro.test

import kotlinx.coroutines.*

fun main() = runBlocking {
    launch {
        println("launch1 started ${Thread.currentThread().name}")
        delay(3000)
        println("launch1 ended ${Thread.currentThread().name}")
    }


    launch(Dispatchers.Default){
        println("launch2 started ${Thread.currentThread().name}")
        delay(2000)
        println("launch2 ended ${Thread.currentThread().name}")
    }
    println("go go ${Thread.currentThread().name}")
}

还是先看正确答案吧:
go go main
launch2 started DefaultDispatcher-worker-1
launch1 started main
launch2 ended DefaultDispatcher-worker-1
launch1 ended main
你应该已经发现了,这个是上个实例 代码就有一点点区别,对,就是因为这个区别导致了和前面实例的输出顺序不一样,就是它

launch(Dispatchers.Default)

那这玩意是干嘛的呢,为啥它会导致上面的输出结果呢?
首先我们要理解,协程都有宿主线程,官网管这个线程叫做 underlying thread, 我们简称它为ut。 所谓的协程挂起,其实就是在ut上做个登记,这个登记记住了如何恢复这个挂起的协程,也就是所谓的挂起点, 此时另外的等待挂起点协程就被唤醒。 说白了,就是协程要想执行 得等待ut有挂起点或者说空闲,当然了ut最开始是空闲的
那么这个代码launch(Dispatchers.Default)就是指明了使用的ut是一个共享线程池 最少2,最多=cpu数, 这个看官网好了。 那么明白了这个原理后就好分析了,那么我们看出这里面有两个宿主线程 utMain 和 utDefault。有3个协程 #main, #1, #2
#main 和 #1的宿主线程是 utMain
#2 宿主线程是utDefault
最先得到调度的是协程 #main,这显而易见哈

    launch(Dispatchers.Default){
        println("launch2 started ${Thread.currentThread().name}")
        delay(2000)
        println("launch2 ended ${Thread.currentThread().name}")
    }

这个#2 也的确是刚一创建就处于唤醒状态,因为utDefault并没被其它协程“占用”,但凡事有先来后到 我#main正在执行计算 #2您请稍后, 所以“go go”就先出来了, #main挂起, #1得以唤醒,但你别忘了 我#2早就唤醒了哦,所以#2先执行,输出了"launch2 started", 说到这之后,剩下的都是显而易见的。
好了,我们循序渐进慢慢来,到此为止

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容