我们要说明一下,这里的实例分析并不会去分析协程的源码,所以说这个是个使用层面的分析。因为我本来就是个使用者,不太关心它是怎么实现的, 我们此次要进一步了解协程的执行顺序,调度原理,还是先出题,说出下面实例的输出顺序,如能说出代表你对协程已经达到入门级,不能的话看第一个实例
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", 说到这之后,剩下的都是显而易见的。
好了,我们循序渐进慢慢来,到此为止