协程&挂起&恢复

协程

协程通用概念:Coroutination passing style(CPS),即传接续体风格。

kotlin协程可以简化为【状态机+续体CPS】

  • 续体负责传递上一步骤的执行结果和后续操作步骤
  • 状态机负责记录和转移执行状态(这里值执行点位置)

挂起

一段协程的代码,例如通过launch启动一个协程,内部的代码,如果存在挂起点,那么就会以挂起点为分界把代码包装为一个续体类,也就是协程CPS转换。

在kotlin中,协程CPS本质也确实是做了一个callback的处理(在编译期)。

在上面的【状态机+续体CPS】 模型中,挂起点之前的代码,就是状态机状态转移前执行,随后把挂起点后的代码包装为续体coroutination等待恢复后继续执行。 这里在包装续体时,会将当前续体以及状态一起传递给新的续体(或称子续体)

恢复

恢复机制:挂起点通知续体恢复。例如我们常用的suspendCoroutine{}, 在挂起点通过调用续体的resume/resumeWith来通知恢复。

kotlin 复制代码
suspend fun doSomething(): String = suspendCoroutine {
    net.getDataFromRemote() { data ->
        it.resume(data)
    }
}

Q:遇到挂起点了,不是已经被挂起了吗....为什么挂起点还能通知恢复?是什么机制 ?

A:

  • 挂起点也是另一个执行点,如果把协程简化为线程池中的runnable,那么挂起点内部也是一个runnable,只不过给它传递了父协程,然后在内部runnable执行结束时,调用父runnable的resumeWith
  • 这也是为什么协程切换可能会发生在同一个线程上的原因
  • 能恢复父协程继续往下执行的原因也在于,在挂起之前执行状态会转移到下一个状态,因此在恢复时就可以根据这个状态找到对应的恢复点继续执行

和线程对比

协程也常常被称为轻量级线程。

线程:

  • 操作系统底层支持的,CPU执行的最小单元
  • 线程间内存空间是独立的,线程切换需要较大的系统开销
    • 上下文保存和恢复,涉及了用户态和内核态的切换
  • 启动一个线程执行任务,那么线程就会被这个任务阻塞;
  • 由于资源占用和系统开销,线程开启的数量往往是有限的、较少的

协程:

  • 协程是上层设计,不同的开发语言有不同的实现
  • 不同的协程是共享线程资源的,协程间的切换就不涉及太多的系统开销,也不涉及用户态和内核态的切换
  • 协程是非阻塞式挂起,即不阻塞线程的,因此即使在安卓开发中也可以放心使用
  • 协程可以同时启动很多,甚至是数以万计

你可参考:

相关推荐
雨季66614 分钟前
Flutter 三端应用实战:OpenHarmony “专注时光盒”——在碎片洪流中守护心流的数字容器
开发语言·前端·安全·flutter·交互
tao35566732 分钟前
【用AI学前端】HTML-02-HTML 常用标签(基础)
前端·html
2601_9495328435 分钟前
Psello HTML Template: A Developer‘s Deep-Dive Review and Guide - Download Free
前端·windows·html·seo·wordpress·gpl
CappuccinoRose36 分钟前
CSS前端布局总指南
前端·css·学习·布局·flex布局·grid布局·float布局
穿过锁扣的风1 小时前
如何操作HTML网页
前端·javascript·html
San30.1 小时前
从零构建坚固的前端堡垒:TypeScript 与 React 实战深度指南
前端·react.js·typescript
yunhuibin1 小时前
VideoPipe环境搭建及编译ubuntu240403
前端·人工智能
CHANG_THE_WORLD2 小时前
PDF文档结构分析 一
前端·pdf
东东5162 小时前
果园预售系统的设计与实现spingboot+vue
前端·javascript·vue.js·spring boot·个人开发
rainbow68892 小时前
Python学生管理系统:JSON持久化实战
java·前端·python