kotlin 协程(Coroutine)

Coroutine(协程)的转换原理:

在 kotlin 中,Coroution 是一种轻量级的线程管理方式,其转换原理涉及 状态机生成挂起函数转换调度器机制

一、协程的本质:状态机

kotlin 协程通过 编译器生成状态机 实现。(当你编写一个挂起函数(suspend))或协程体时,编译器会将其转换为一个状态机类。

示例代码

kotlin

suspend fun fetchData() {val data = loadFromNetwork() // 挂起点 1processData(data)          // 挂起点 2
}
编译后等价代码(简化版)

java

// 编译器生成的状态机类
final class FetchDataKt$fetchData$1 extends SuspendLambda implements Function2<Unit, Continuation<? super Unit>, Object> {int label;                // 当前状态Object result;            // 中间结果String data;              // 局部变量FetchDataKt$fetchData$1(Continuation<? super Unit> completion) {super(2, completion);}@Overridepublic final Object invokeSuspend(Object $result) {this.result = $result;this.label |= Integer.MIN_VALUE;// 根据状态跳转到不同代码段switch (label) {case 0: // 初始状态// 执行 loadFromNetwork()return loadFromNetwork(this);case 1: // 恢复状态 1data = (String) result;processData(data);return Unit.INSTANCE;default:throw new IllegalStateException("call to 'resume' before 'invoke' with coroutine");}}
}

二、挂起函数的转换

挂起函数(suspend)的关键在于 保存和恢复执行状态

  1. 挂起点(Suspension Point):
    当协程遇到挂起函数(如 delay()withContext())时,会:

    • 保存当前状态(局部变量、执行位置)到状态机。
    • 返回到调用者(不会阻塞线程)。
  2. 恢复执行
    当挂起条件满足(如网络请求完成)时,通过 Continuation.resume() 恢复状态机:

    • 恢复局部变量和执行位置。
    • 从挂起点继续执行。

三、协程调度器(CoroutineDispatcher)

协程通过 调度器 决定在哪个线程上执行:

常见调度器
  • Dispatchers.Main:Android 主线程,用于 UI 操作。
  • Dispatchers.IO:IO 优化的线程池,适合网络 / 文件操作。
  • Dispatchers.Default:CPU 密集型任务的默认线程池。
  • newSingleThreadContext():创建专用单线程。
调度器工作原理
  1. 协程启动

    GlobalScope.launch(Dispatchers.IO) {val data = fetchData() // 在 IO 线程执行withContext(Dispatchers.Main) {updateUI(data)     // 切换到主线程}
    }
    
  2. 线程切换实现

  • withContext() 是一个挂起函数,会:
  1. 保存当前状态。
  2. 将任务提交到目标调度器的线程。
  3. 在新线程恢复执行。

四、协程与线程的关系

特性协程线程
创建成本极低(约 2KB 内存)高(约 1MB 栈空间)
调度方式协作式(由协程自己决定何时挂起)抢占式(由操作系统调度)
切换开销极小(仅状态机跳转)高(上下文切换涉及内核操作)
数量限制可创建数百万个通常限制在数千个
阻塞影响仅阻塞当前协程阻塞整个线程

五、关键组件与机制

1. Continuation
  • 协程的核心接口,定义了恢复执行的方法:
    interface Continuation<in T> {val context: CoroutineContextfun resumeWith(result: Result<T>)
    }
    
  • 编译器会为每个协程生成 Continuation 的实现。
2. Job
  • 协程的生命周期控制器,可用于取消、检查状态:
    val job = launch { ... }
    job.cancel() // 取消协程
    
3. CoroutineContext
  • 存储协程的上下文信息(如调度器、异常处理器):
    val scope = CoroutineScope(Dispatchers.IO + Job())
    

      

六、优化与调试建议

  1. 避免阻塞调度器线程

    // 错误:在 IO 调度器中执行 CPU 密集型任务
    withContext(Dispatchers.IO) {heavyCalculation() // 应使用 Dispatchers.Default
    }
    
  2. 使用协程作用域(CoroutineScope)
    避免内存泄漏,自动管理协程生命周期:

    class MyViewModel : ViewModel() {fun fetchData() = viewModelScope.launch { ... }
    }
    
  3. 调试工具

    • 使用 runBlocking { ... } 在测试中阻塞主线程。
    • 通过 LoggingInterceptor 记录协程调度过程。

七、总结

Kotlin 协程通过 状态机转换 和 非阻塞挂起机制,实现了高效的线程管理:

  • 编译器将协程代码转换为状态机,保存执行状态。
  • 调度器决定协程在哪个线程执行,支持灵活切换。
  • 相比传统线程,协程大幅降低资源消耗,提升并发能力。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/86708.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/86708.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

线性变换之维数公式(秩-零化度定理)

秩数-零化度定理(rank-nullity theorem) 目录 1. (映射)零空间(线性映射或变换的核)(null-space或nullspace) 2. 跨度(或开度)(span) 3. (线性映射的)零化度(nullity) 4. 线性变换的维数公式(秩数-零化度定理)(rank-nullity theorem) 5. 函数的上域(codomain) 1…

Spring Cloud Gateway 实战:网关配置与 Sentinel 限流详解

Spring Cloud Gateway 实战&#xff1a;网关配置与 Sentinel 限流详解 在微服务架构中&#xff0c;网关扮演着统一入口、负载均衡、安全认证、限流等多种角色。Spring Cloud Gateway 是 Spring Cloud 官方推出的新一代网关组件&#xff0c;相比于第一代 Netflix Zuul&#xff…

JAVA-常用API(二)

目录 1.Arrays 1.1认识Arrays 1.2Arrays的排序 2.JDK8的新特性&#xff1a;Lambda表达式 2.1认识Lambda表达式 2.2用Lambda表达式简化代码、省略规则 3.JDK8的新特性&#xff1a;方法引用&#xff08;进一步简化Lambda表达式&#xff09; 3.1 静态方法引用 3.2 实例方法引…

深入理解PHP的命名空间

命名空间是PHP 5.3引入的一个特性&#xff0c;它的主要目的是解决在大型应用程序中可能出现的名称冲突问题。在没有命名空间的情况下&#xff0c;如果两个不同的库或模块定义了相同名称的函数或类&#xff0c;那么在使用这些库或模块的时候就会引发冲突。为了解决这个问题&…

SwiftUI学习笔记day5:Lecture 5 Stanford CS193p 2023

SwiftUI学习笔记day5:Lecture 5 Stanford CS193p 2023 课程链接&#xff1a;Lecture 5 Stanford CS193p 2023代码仓库&#xff1a;iOS课程大纲&#xff1a; Enum 定义&#xff1a;enum MyType { … }关联值&#xff1a;case drink(name: String, oz: Int)匹配&#xff1a;switc…

idea 报错:java: 非法字符: ‘\ufeff‘

idea 报错&#xff1a;java: 非法字符: ‘\ufeff‘ 解决方案&#xff1a;

数据结构与算法之美:图

Hello大家好&#xff01;很高兴我们又见面啦&#xff01;给生活添点passion&#xff0c;开始今天的编程之路&#xff01; 我的博客&#xff1a;<但凡. 我的专栏&#xff1a;《编程之路》、《数据结构与算法之美》、《题海拾贝》、《C修炼之路》 欢迎点赞&#xff0c;关注&am…

SpringBoot -- 热部署

9.SpringBoot 热部署&#xff08;自动重启&#xff09; 在实际开发过程中&#xff0c;每次修改代码就得将项目重启&#xff0c;重新部署&#xff0c;对于一些大型应用来说&#xff0c;重启时间需要花费大量的时间成本。对于一个后端开发者来说&#xff0c;重启过程确实很难受啊…

HarmonyOS 5浏览器引擎对WebGL 2.0的支持如何?

以下是HarmonyOS 5浏览器引擎对‌WebGL 2.0‌支持的详细技术分析&#xff1a; 一、核心支持能力 ‌系统能力声明 HarmonyOS 5 浏览器引擎通过 SystemCapability.Graphic.Graphic2D.WebGL2 提供对 WebGL 2.0 的底层支持 支持的关键特性包括&#xff1a; OpenGL ES 3.0 特性…

Class1线性回归

Class1线性回归 买房预测 要根据历史数据来预测一套房子的价格。你发现影响房价的因素有很多&#xff0c;于是你决定使用线性回归模型来预测房价。 影响房价的因素如下&#xff1a; 房屋面积&#xff08;平方米&#xff09; 房龄&#xff08;年&#xff09; 离地铁站的距离&a…

Vue.js 3:重新定义前端开发的进化之路

Vue.js 3&#xff1a;重新定义前端开发的进化之路 引言&#xff1a;一场酝酿已久的革新 2020年9月18日&#xff0c;Vue.js团队以代号"One Piece"正式发布3.0版本&#xff0c;这不仅是框架发展史上的重要里程碑&#xff0c;更是前端工程化领域的一次革命性突破。历经…

Unity性能优化-渲染模块(1)-CPU侧(1)-优化方向

Unity 中渲染方面的优化大致可以划分为以下几块核心内容&#xff1a; CPU 优化 (减少 Draw Calls 和 CPU 瓶颈) GPU 优化 (减少像素着色和 GPU 瓶颈) 内存和显存优化 (Resource Management) 光照优化 (Lighting & Global Illumination) 这四个方面是相互关联的。一个方…

AI矢量图与视频无痕修复:用Illustrator与After Effects解锁创作新维度

最近因一个项目&#xff0c;有机会深度体验了奥地利Blueskyy艺术学院授权的Adobe教育版全家桶&#xff0c;过程中发现了不少令人惊喜的“黑科技”&#xff0c;很想和大家分享这份发掘宝藏的喜悦。一句话总结这次体验&#xff1a;慷慨且稳定。比如&#xff0c;它每周提供高达150…

Maven Javadoc 插件使用详解

Maven Javadoc 插件使用详解 maven-javadoc-plugin 是 Maven 项目中用于生成 Java API 文档的标准插件&#xff0c;它封装了 JDK 的 javadoc 工具&#xff0c;提供了更便捷的配置和集成方式。 一、基本使用 1. 快速生成 Javadoc 在项目根目录执行以下命令&#xff1a; bas…

Apache Kafka 面试应答指南

Apache Kafka 核心知识详解与面试应答指南 一、Apache Kafka 概述 Apache Kafka 作为一款分布式流处理框架,在实时构建流处理应用领域发挥着关键作用。其最广为人知的核心功能,便是作为企业级消息引擎被众多企业采用。 二、消费者组 (一)定义与原理 消费者组是 Kafka 独…

在NVIDIA Jetson和RTX上运行Google DeepMind的Gemma 3N:多模态AI的边缘计算革命

在NVIDIA Jetson和RTX上运行Google DeepMind的Gemma 3N&#xff1a;多模态AI的边缘计算革命 文章目录 在NVIDIA Jetson和RTX上运行Google DeepMind的Gemma 3N&#xff1a;多模态AI的边缘计算革命引言&#xff1a;多模态AI进入边缘计算时代文章结构概览 第一章&#xff1a;Gemma…

iOS打包流程中的安全处理实践:集成IPA混淆保护的自动化方案

随着iOS应用上线节奏的加快&#xff0c;如何在持续集成&#xff08;CI&#xff09;或交付流程中嵌入安全处理手段&#xff0c;成为开发团队构建自动化发布链路时不可忽视的一环。特别是在App已经完成构建打包&#xff0c;准备分发前这一阶段&#xff0c;对IPA进行结构层面的加固…

FFmpeg进行简单的视频编辑与代码写法实例

使用 FFmpeg 进行简单的视频编辑非常强大。它是一个命令行工具&#xff0c;虽然一开始可能看起来有点复杂&#xff0c;但掌握了基本命令后会非常有用。 以下是一些常见的简单视频编辑操作及其 FFmpeg 命令&#xff1a; 1. 剪切视频 如果你想从一个视频中剪切出一段&#xff0…

如何使用免费软件写论文?六个免费论文生成软件使用指南

在学术写作中&#xff0c;利用AI技术和免费的写作工具可以极大地提高效率&#xff0c;尤其对于需要处理大量文献、结构化写作的论文来说&#xff0c;使用合适的软件能节省时间&#xff0c;提升论文质量。这里为您推荐六个免费的论文生成软件&#xff0c;并提供使用指南&#xf…

大数据系统架构实践(二):Hadoop集群部署

大数据系统架构实践&#xff08;二&#xff09;&#xff1a;Hadoop集群部署 文章目录 大数据系统架构实践&#xff08;二&#xff09;&#xff1a;Hadoop集群部署一、Hadoop简介二、部署前准备三、部署Hadoop集群1. 下载并解压安装包2. 配置hadoop-env.sh3. 配置core-site.xml4…