Android面试题之Kotlin 协程的挂起、执行和恢复过程

简介: 了解Kotlin协程的挂起、执行和恢复机制。挂起时,状态和上下文(局部变量、调用栈、调度器等)被保存;挂起点通过`Continuation`对象处理,释放线程控制权。当恢复条件满足,调度器重新分配线程,调用`resumeWith`恢复执行。关注公众号“AntDream”获取更多并发知识。

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

协程的挂起、执行和恢复过程到底是怎么样的?

协程(Coroutine)的挂起和恢复机制是其高效管理并发性的核心。这些过程涉及多个关键步骤,包括状态和上下文的保存、释放线程控制权、以及恢复时的通知等。
我们来详细讲解一下这些机制。

1. 协程挂起和恢复时保存的状态和上下文

1.1 状态信息

协程的状态信息主要包括:

  • 局部变量:函数当前执行到的位置以及所有局部变量的值。
  • 挂起点:协程挂起的位置,这个位置通常是代码中的一个挂起点(suspend函数)。
  • 调用栈:它对应当前执行的协程堆栈帧,可以看作是对函数调用链的保存。

1.2 上下文信息

协程的上下文信息通常包括:

  • 调度器:即协程运行的调度器(例如 Dispatchers.IO, Dispatchers.MainDispatchers.Default)。
  • 异步工作器:包含了协程的执行环境和工作状态。
  • 取消状态:协程是否被取消或处于取消状态。

2. 状态与上下文的保存形式

2.1 协程堆栈帧

协程在挂起时,会将当前的堆栈帧转换为对象并存储在堆中。这个对象包含了所有当前帧的局部变量、挂起点以及其他必要信息。恢复时,这个对象重新转换为堆栈帧并继续执行。

2.2 Continuation

Kotlin中的挂起函数实质上会被编译器转换成带有回调的 Continuation 对象。该对象包含两个主要部分:

  • 上下文(Continuation Context):绑定的执行环境。
  • 恢复逻辑(Resume Logic):保存和处理挂起点的逻辑。
interface Continuation<in T> {
    val context: CoroutineContext
    fun resumeWith(result: Result<T>)
}

3. 协程挂起后切换到其他线程执行

当协程遇到挂起点(如 delay, awaitsuspend 函数)时,它会触发挂起机制,具体步骤如下:

3.1 挂起点的处理

当协程在挂起点被挂起时,当前函数状态和局部变量会被保存到 Continuation 对象中。
挂起函数会将 Continuation 对象传递给协程的调度器。

3.2 挂起协程和释放线程

调度器会暂停当前协程的执行,把线程控制权交给调度器管理的线程池或其他任务,从而释放当前线程。

4. 切换到其他线程执行

4.1 恢复协程条件满足

当挂起的条件满足(例如 delay 到期,或者异步任务完成),调度器会收到执行恢复逻辑的通知。

4.2 调度器分配新的线程运行

调度器会将保存的 Continuation 对象重新分配给线程池中的某个线程,调用 resumeWith 方法恢复协程:

continuation.resumeWith(Result.success(result))

这时,协程会恢复到挂起点之后的代码继续正常执行。

5. 协程在其他线程执行完后的通知机制

5.1 异步任务完成通知

当协程在新的线程中执行完任务(比如完成网络请求等异步任务)时,执行环境会调用 ContinuationresumeWith 方法:

continuation.resumeWith(Result.success(result))

5.2 通知调度器

resumeWith 方法会触发协程恢复处理,同时通知调度器该协程已准备好继续执行。

6. 调度器恢复协程到原线程

6.1 检查恢复环境

调度器检查协程需要恢复的环境,特别是上下文中的线程调度信息。如果协程需要恢复到特定线程(例如主线程),调度器会安排该任务。

6.2 分配线程

调度器找到或分配合适的线程,根据协程上下文完成恢复调度。典型的调度器如 Dispatchers.Main 或自定义调度器负责将任务放回特定线程运行。

6.3 恢复执行

调度器调用 Continuationresume 方法,将保存的上下文和状态恢复到协程堆栈:

continuation.resume(result)

然后,协程在新的或原来的线程上恢复执行挂起点之后的代码。
以下是一个简单的示例,展示了协程如何在挂起后切换到不同线程并恢复到主线程:

import kotlinx.coroutines.*

fun main() = runBlocking {
    withContext(Dispatchers.Main) {
        println("Running on main thread: ${Thread.currentThread().name}")

        withContext(Dispatchers.IO) {
            println("Switching to IO thread: ${Thread.currentThread().name}")
            delay(1000) // 挂起点
            println("Resuming on IO thread: ${Thread.currentThread().name}")
        }

        println("Back to main thread: ${Thread.currentThread().name}")
    }
}

在这个例子中:

  • withContext(Dispatchers.Main) 确保代码一开始运行在主线程。
  • withContext(Dispatchers.IO) 切换到 I/O 线程,执行 delay 挂起。
  • 挂起后,协程会保存当前状态和上下文,并交由 Dispatchers.IO 管理。
  • 延时结束后,Dispatchers.IO 恢复协程执行。
  • withContext(Dispatchers.Main) 之后的代码切换回主线程,保证恢复到原来的执行环境。

总结

综上所述,Kotlin 协程在挂起和恢复过程中,通过调度器实现线程的切换和任务调度:

  1. 挂起点保存当前状态由Continuation管理。
  2. 调度器控制挂起后的线程控制权交还。
  3. 恢复条件满足时,调度器分配新线程,调用resumeWith方法恢复协程。
  4. 调度器检查恢复环境,分配合适线程并调用resume方法继续执行。

欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
23小时前
|
安全 JavaScript 前端开发
kotlin开发安卓app,JetPack Compose框架,给webview新增一个按钮,点击刷新网页
在Kotlin中开发Android应用,使用Jetpack Compose框架时,可以通过添加一个按钮到TopAppBar来实现WebView页面的刷新功能。按钮位于右上角,点击后调用`webViewState?.reload()`来刷新网页内容。以下是代码摘要:
|
1天前
|
网络协议 算法 安全
小米安卓春招面试一面
小米安卓春招面试一面
17 3
|
1天前
|
缓存 网络协议 Java
Android面试题之Java网络通信基础知识
Socket是应用与TCP/IP通信的接口,封装了底层细节。网络通信涉及连接、读写数据。BIO是同步阻塞,NIO支持多路复用(如Selector),AIO在某些平台提供异步非阻塞服务。BIO示例中,服务端用固定线程池处理客户端请求,客户端发起连接并读写数据。NIO的关键是Selector监控多个通道的事件,减少线程消耗。书中推荐《Java网络编程》和《UNIX网络编程》。关注公众号AntDream了解更多。
11 2
|
2天前
|
XML JSON Java
Android面试题 之 网络通信基础面试题
序列化对比:Serializable码流大、性能低;XML人机可读但复杂;JSON轻量、兼容性好但空间消耗大;ProtoBuff高效紧凑。支持大量长连接涉及系统限制调整、缓冲区优化。select/poll/epoll是IO多路复用,epoll在高连接数下性能更优且支持边缘触发。水平触发持续通知数据,边缘触发仅通知新数据。直接内存减少一次拷贝,零拷贝技术如sendfile和MMAP提升效率。关注公众号&quot;AntDream&quot;了解更多技术细节。
7 1
|
2天前
|
Android开发 Kotlin
Android面试题 之 Kotlin DataBinding 图片加载和绑定RecyclerView
本文介绍了如何在Android中使用DataBinding和BindingAdapter。示例展示了如何创建`MyBindingAdapter`,包含一个`setImage`方法来设置ImageView的图片。布局文件使用`&lt;data&gt;`标签定义变量,并通过`app:image`调用BindingAdapter。在Activity中设置变量值传递给Adapter处理。此外,还展示了如何在RecyclerView的Adapter中使用DataBinding,如`MyAdapter`,在子布局`item.xml`中绑定User对象到视图。关注公众号AntDream阅读更多内容。
9 1
|
2天前
|
JavaScript Java Android开发
kotlin安卓在Jetpack Compose 框架下跨组件通讯EventBus
**EventBus** 是一个Android事件总线库,简化组件间通信。要使用它,首先在Gradle中添加依赖`implementation &#39;org.greenrobot:eventbus:3.3.1&#39;`。然后,可选地定义事件类如`MessageEvent`。在活动或Fragment的`onCreate`中注册订阅者,在`onDestroy`中反注册。通过`@Subscribe`注解方法处理事件,如`onMessageEvent`。发送事件使用`EventBus.getDefault().post()`。
|
2天前
|
JavaScript 前端开发 Android开发
kotlin安卓在Jetpack Compose 框架下使用webview , 网页中的JavaScript代码如何与native交互
在Jetpack Compose中使用Kotlin创建Webview组件,设置JavaScript交互:`@Composable`函数`ComposableWebView`加载网页并启用JavaScript。通过`addJavascriptInterface`添加`WebAppInterface`类,允许JavaScript调用Android方法如播放音频。当页面加载完成时,执行`onWebViewReady`回调。
|
4天前
|
Android开发
Android面试题之activity启动流程
该文探讨了Android应用启动和Activity管理服务(AMS)的工作原理。从Launcher启动应用开始,涉及Binder机制、AMS回调、进程创建、Application和Activity的生命周期。文中详细阐述了AMS处理流程,包括创建ClassLoader、加载APK、启动Activity的步骤,以及权限校验和启动模式判断。此外,还补充了activity启动流程中AMS的部分细节。欲了解更多内容,可关注公众号“AntDream”。
9 1
|
4天前
|
vr&ar 数据库 Android开发
Android面试题之ActivityManagerService的启动流程
本文探讨了Android系统的SystemServer启动过程,包括创建SystemContext、引导服务、启动各类核心服务以及AMS的启动和初始化。AMS负责管理activity、广播队列、provider等,并设置SystemProcess,安装系统Provider。当AMS调用SystemReady时,系统UI准备启动,启动Launcher。文中还对比了init、zygote和system_server进程的角色。最后推荐了两本关于Android内核剖析的书籍:柯元旦教授的《Android内核剖析》和罗升阳的《Android系统源代码情景分析》。关注公众号AntDream获取更多内容。
6 0
|
5天前
|
JSON 安全 调度
Android面试题之Kotlin协程一文搞定
本文介绍了协程的基础知识,强调它是轻量级线程,用于处理耗时任务而不阻塞主线程,确保主线程安全。协程特点包括使异步逻辑同步化,并允许函数挂起和恢复。挂起函数由`suspend`关键字标识,只能在协程内部调用。挂起与阻塞的主要区别在于挂起不会导致主线程ANR。 结构化并发和协程作用域(如`CoroutineScope`、`GlobalScope`、`MainScope`等)提供了任务管理,文章还探讨了并发、启动模式、协程取消、超时任务以及资源释放等主题。
13 0