本文主要是介绍Kotlin: Suspend挂起,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
1.协程是什么?
线程框架。【更方便】
协程就是launch里面的代码。
2.挂起谁?
挂起协程。
launch创建的协程在执行到某一个suspend函数挂起函数的时候,这个协程会被suspend(被挂起)
3.从哪儿挂起?
从当前线程挂起。
这个协程从正在执行它的线程上脱离了。不是这个协程停下来了而是协程所在的线程从这行代码开始不再运行这个协程了。
线程和协程分2波走了。
4.分离后的线程做什么?
协程的代码块在线程里到了suspend函数的时候突然执行完了、返回了。然后线程就做它自己的事情了。比如这个线程是后台线程,那么这个线程就nothing to do.该回收回收,该利用利用。如果这个线程是main thread,则线程会回去继续工作。
图中的伪代码是启动了一个执行在主线程的协程。它实质上会在主线程post()一个new task:
handler.post{val todo = suspendToDo()...
}
这个task就是协程代码。当这个协程被挂起的时候,实质上就是post()的这个task提前结束了。
然后主线程继续刷新ui。
协程还没用执行完,剩下的代码:协程。
5.分离后的协程做什么?
函数的代码在到达suspend函数的时候被掐断了,它会从这个挂起函数开始继续往下执行【是在指定的线程(挂起函数指定的)】
launch(Dispatchers.Main){...val todo=suspendToDo()....
}suspend fun suspendToDo(){withContext(Dispatchers.IO){....}
}
例如上面就是函数内部的withContext()所指定的io线程。
挂起函数之后,协程为我们做的事:自动把线程再切回来。
例如上面伪代码中,协程原来是在main thread中运行的,切回来就是在挂起函数执行完成之后,协程会再post()一个task,让剩下的代码继续回到主线程去执行。这是因为Dispatchers调度器可以指定协程执行的线程还可以在suspend挂起函数之后自动再切回来。【设置特殊的调度器可以让挂起函数执行完后不再切回来】
挂起的定位: 暂时切走,稍后再切回来。
所以,挂起:
协程在执行到有supend标记的函数的时候会被挂起,挂起和开启一个协程一样,就是切个线程。只不过挂起函数在执行完毕之后协程会自动的重新切回它原先的那个线程。挂起就是一个稍后会被自动切回来的线程切换。
切回来的动作,叫:resume(恢复)
挂起之后是需要恢复的,恢复这个功能是协程的。挂起函数不在协程里被调用,那么这个恢复的功能就没法实现了。
怎么做到挂起的?
如果单纯的定义一个suspend标识的函数然后在协程里面去调用它是没效果的。因为它不知道往哪里切。
对于自定义的挂起函数,例如withContext()有调度器。
协程的被挂起并不是发生在外部这个挂起函数被调用的时候,而是里面的挂起函数被调用的时候,例如常见的withContext()。【withcotext()也不是真的切线程的点,而是其内部的底层代码】
所以supend关键字并不启到协程挂起/切换线程的作用。真要挂起协程还需要在挂起函数里去调用另一个挂起函数【需要是协程自带的、内部实现了协程挂起代码的。或它的内部直接或者间接调用了某一个挂起函数,让它去真正的挂起。】
对于自定义挂起函数,需要在该函数内部直接/间接调用到某一个自带的挂起函数才行【例如withContext()】
Suspend关键字到底什么作用?
提醒。【函数的创建者对函数的调用者的提醒】
“我”是一个耗时的函数,我被我的创建者用挂起的方式放到了后台运行,所以要在协程里调用“我”。
它让我们的主线程不卡。通过挂起函数这种形式把耗时task切线程的这个工作交给了函数的创建者而不是调用者。
调用者只会收到一个提醒:你只需要把我放在一个协程里调用。
【挂起操作靠的是挂起函数里的实际代码,而不是关键字】
总结:
怎么custom a supend function?
1.什么时候自定义?
- 原则:耗时【io操作和计算操作】(特殊情况:事情做起来不慢,但是需要等待(delay))
2.怎么写?
- 加上suspend关键字、用withcontext()把函数的内容包住。
学习扔物线大佬的suspend总结。
这篇关于Kotlin: Suspend挂起的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!