Kotlin中協(xié)程的創(chuàng)建過程詳析
為什么需要協(xié)程?
協(xié)程可以簡化異步編程,可以順序地表達(dá)程序,協(xié)程也提供了一種避免阻塞線程并用更廉價、更可控的操作替代線程阻塞的方法 – 掛起函數(shù)。
Kotlin 的協(xié)程是依靠編譯器實現(xiàn)的, 并不需要操作系統(tǒng)和硬件的支持。編譯器為了讓開發(fā)者編寫代碼更簡單方便, 提供了一些關(guān)鍵字(例如suspend), 并在內(nèi)部自動生成了一些支持型的代碼。
創(chuàng)建并啟動協(xié)程
fun create.main() { //1. 創(chuàng)建協(xié)程體 val coroutine = suspend { println("in coroutine") 5 }.createCoroutine(object: Continuation<Int> { override fun resumeWith(result: Result<Int>) { println("coroutine end: $result") } override val context: CoroutineContext get() = EmptyCoroutineContext }) //2. 執(zhí)行協(xié)程 coroutine.resume(Unit) }
上面代碼的輸出結(jié)果:
in coroutine coroutine end: Success(5)
協(xié)程的執(zhí)行過程
調(diào)用棧流程如下
- 我們通過 suspend block#createCoroutine 得到的 coroutine 實際是 SafeContinuation 對象
- SafeContinuation 實際上是代理類,其中的 delegate 屬性才是真正的 Continuation 對象
- suspend block 中的代碼在 BaseContinuationImpl 中執(zhí)行
- 我們的匿名內(nèi)部類對象 Continuation 被回調(diào)
suspend block 是如何變?yōu)閰f(xié)程體被執(zhí)行的?
我們分析調(diào)用棧得知,resumeWith 最終是在 BaseContinuationImpl 中執(zhí)行的,下面來看看代碼
@SinceKotlin("1.3") internal abstract class BaseContinuationImpl( public val completion: Continuation<Any?>? ) : Continuation<Any?>, CoroutineStackFrame, Serializable { public final override fun resumeWith(result: Result<Any?>) { var current = this var param = result while (true) { probeCoroutineResumed(current) with(current) { val completion = completion!! val outcome: Result<Any?> = try { val outcome = invokeSuspend(param) //1.這里執(zhí)行了 suspend block if (outcome === COROUTINE_SUSPENDED) return Result.success(outcome) } catch (exception: Throwable) { Result.failure(exception) } releaseIntercepted() if (completion is BaseContinuationImpl) { current = completion param = outcome } else { completion.resumeWith(outcome) //2.這里回調(diào)了我們的匿名內(nèi)部類 return } } } } protected abstract fun invokeSuspend(result: Result<Any?>): Any? //3. 抽象方法 }
在代碼注釋 1. 處,調(diào)用 current.invokeSuspend,執(zhí)行了我們定義的協(xié)程體,證明 suspend block 其實是 BaseContinuationImpl 的子類
在 2. 處,協(xié)程體執(zhí)行完畢后,我們的代碼收到了完成回調(diào)
在 3. 處,可以發(fā)現(xiàn) invokeSuspend 是個抽象方法,suspend block 就是這個方法的具體實現(xiàn)
下面我通過斷點,進(jìn)一步分析 suspend block 是通過哪個子類執(zhí)行的。
可以看到 current 是名為 {文件}${方法}${變量}$1 格式的對象,證明 kotlin 編譯器遇到 suspend 關(guān)鍵字后會幫我們生成一個 BaseContinuationImpl 的子類
那么,這個子類到底是什么呢?將 kt 編譯為 .class 再通過 jadx 打開后,得到的 java 代碼如下
public final class CreateCoroutineKt { public static final void create.main() { Continuation coroutine = ContinuationKt.createCoroutine(new CreateCoroutineKt$create.main$coroutine$1(null), new CreateCoroutineKt$create.main$coroutine$2()); Unit unit = Unit.INSTANCE; Result.Companion companion = Result.Companion; coroutine.resumeWith(Result.constructor-impl(unit)); } }
final class CreateCoroutineKt$create.main$coroutine$1 extends SuspendLambda implements Function1<Continuation<? super Integer>, Object> { int label; CreateCoroutineKt$create.main$coroutine$1(Continuation<? super CreateCoroutineKt$create.main$coroutine$1> continuation) { super(1, continuation); } @NotNull public final Continuation<Unit> create(@NotNull Continuation<?> continuation) { return new CreateCoroutineKt$create.main$coroutine$1(continuation); } @Nullable public final Object invoke(@Nullable Continuation<? super Integer> continuation) { return create(continuation).invokeSuspend(Unit.INSTANCE); } @Nullable public final Object invokeSuspend(@NotNull Object obj) { IntrinsicsKt.getCOROUTINE_SUSPENDED(); switch (this.label) { case 0: ResultKt.throwOnFailure(obj); System.out.println((Object) "in coroutine"); //協(xié)程體的邏輯 return Boxing.boxInt(5); default: throw new IllegalStateException("call to 'resume' before 'invoke' with coroutine"); } } }
明顯看出,kt 編譯器幫助我們把 suspend 關(guān)鍵字變?yōu)榱?SuspendLambda 的 子類,并重寫了 invokeSuspend 方法,不難猜出 SuspendLambda 繼承自 BaseContinuationImp
總結(jié)
用一個類圖簡單的總結(jié)一個協(xié)程創(chuàng)建并執(zhí)行的過程。
suspend block(lambda) 在編譯時會轉(zhuǎn)變?yōu)?SuspendLambda 的匿名子類,并把 block 中的邏輯通過重寫 invokeSuspend 實現(xiàn)
調(diào)用 suspend_lambda.createCoroutine 會得到 SafeContinuation 對象,這只是一個代理類,代理的對象正是我們傳入的 SuspendLambda
createCoroutine 的參數(shù)是 completion,代表協(xié)程執(zhí)行完畢的回調(diào)
最終調(diào)用到了 BaseContinuationImpl 的 resumeWith,完成協(xié)程的調(diào)用,調(diào)用完畢的回調(diào)
總結(jié)
到此這篇關(guān)于Kotlin中協(xié)程創(chuàng)建的文章就介紹到這了,更多相關(guān)Kotlin協(xié)程創(chuàng)建內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android 通過代碼設(shè)置、打開wifi熱點及熱點連接的實現(xiàn)代碼
這篇文章主要介紹了Android 通過代碼設(shè)置、打開wifi熱點及熱點連接的實現(xiàn)代碼,本文通過實例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2018-05-05Android實現(xiàn)四級聯(lián)動地址選擇器
這篇文章主要為大家詳細(xì)介紹了Android實現(xiàn)四級聯(lián)動地址選擇器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-10-10Android ViewPager循環(huán)播放廣告實例詳解
這篇文章主要介紹了Android ViewPager循環(huán)播放廣告條實例詳解的相關(guān)資料,需要的朋友可以參考下2017-03-03Android 修改viewpage滑動速度的實現(xiàn)代碼
由于Viewpager的滑動速度是固定的,所以很頭疼,下面小編通過實例代碼給大家分享android 修改viewpage滑動速度的方法,需要的朋友參考下吧2017-09-09Android學(xué)習(xí)筆記之AndroidManifest.xml文件解析(詳解)
這篇文章主要介紹了Android學(xué)習(xí)筆記之AndroidManifest.xml文件解析,需要的朋友可以參考下2015-10-10