kotlin协程[7]:深层揭示runBlocking与coroutineScope之间的异同

在之前写过这样的一个例子,先来回顾一下:


也就是来演示runBlockingcoroutineScope之间的异同点,当时还阐述了一个对它的理论描述,也来回顾一下:

这里再开个篇幅来提出的原因是在于。。这里面有一些深层次的东东需要再次挖掘,而问题的焦点是在:


回到代码根据这段文字的理解照理应该是这样的嘛:

结果肯定不是我们现在所质疑的观点啦,所以这也是需要再单独拎出来值得探讨的东东,这是因为关于runBlockingcoroutineScope是有更加深层的原因的,下面先来阐述一下:

1. runBlocking并非挂起函数;也就是说,调用它的线程会一直位于该函数中,直到协程执行完毕为止。
2. coroutineScope是挂起函数;也就是说,如果其中的协程挂起,那么coroutineScope函数也会挂起。这样,创建coroutineScope的外层函数就可以继续在同一个线程中执行了,该线程会【逃离】coroutineScope之外,并且可以做其他一些事情。

咱们来看一下runBlocking函数的定义:

再来看一下coroutineScope函数的定义:

说实话对于上面的理论描述有点难以理解,怎么最终的welcome的输出是在最后打印的而非咱们预期理解的要立马打印出来,其实需要这样来理解:

所以很明显“welcome”肯定是最后才会被打印出来的,但是!!!貌似上面的有点像是coroutineScope函数阻塞了当前线程,这个观点又与这个理论貌似矛盾了呀:

其实这个理论是没任何问题的,就是理解上需要这样来理解,如下:


如果说coroutineScope是阻塞了当前线程,也就不可能能执行到这句代码:

所以这也能论证coroutineScope确实是不会阻塞当前线程的,而当跳出到runBlocing代码时,它里面会有一个事件循环:


当事件发生时则就会触发事件,也就类似于当休眠完之后就要开始打印语句了,也就相当于事件触发了,如下:

这也就是这句话能打印出来的原因,也就是说,其coroutinScope的真正流程是它会将调度返回给外层runBlocking里面的代码,而且是coroutineScope之上的代码,而非之下的代码,而welcome为啥是最后才打印的真正原因绝对不是因为coroutinScope将线程的代码给阻塞了,这一点确实是比较难理解!!

好,下面了解了这些深层次的理论之后,咱们再以更加正确的姿势来解读一下整个程序的执行流程:

接下来线程就会碰到coroutineScope挂起函数了:

当遇到挂起函数时,就需要立马来区分它之上的代码和之下的代码,记住一点:它之下的代码一定是需要等待coroutineScope中的协程代码整个执行完了才能被执行到【如果这个前提观点不知道,那整个流程就确实是比较难解释了,一定得要有这种概念】,而:

另外一点是当线程遇到了挂起函数会立马从它往上返回,也就是返回到这块代码:



接下来由于延时到了,接着coroutineScope中的这段代码会得到执行:

接着10s过后,里面的协程这块代码就会被打印了:

当这个打印完成,则整个coroutineScope中的协程都执行完了,那该挂起函数也就可以退出了,最后就可以执行挂起函数之下的代码,也就是:

至此!!整个流程就再次以一个全新的视角分析完了~~ 虽说是比较细节,但是对于整个协程的认知理解是非常之重要的!!

你可能感兴趣的:(kotlin协程[7]:深层揭示runBlocking与coroutineScope之间的异同)