当前位置: 首页 > news >正文

Kotlin 协程 (一)

1. Kotlin 协程的核心概念

1.1 协程(Coroutine)
  • 定义:协程是一种轻量级的执行上下文,可以在任何时候挂起和恢复,而不需要阻塞线程。
  • 特点
    • 比传统线程更轻量,开销更小。
    • 支持挂起和恢复,避免了阻塞线程和资源浪费。
    • 提供更简洁的并发编程方式。
1.2 挂起函数(Suspend Function)
  • 定义:使用 suspend 关键字修饰的函数,可以在协程中挂起执行,挂起期间不会阻塞线程。
  • 作用:允许协程在等待任务完成时释放线程资源,待任务完成后恢复执行。
1.3 协程作用域(Coroutine Scope)
  • 定义:协程作用域定义了协程的生命周期,确保协程在作用域结束时被取消。
  • 常见作用域
    • GlobalScope:全局作用域,生命周期与整个应用程序一致(不推荐用于实际开发)。
    • CoroutineScope:自定义作用域,可通过 CoroutineScope(Dispatchers) 创建。
    • runBlocking:用于测试场景,会阻塞当前线程直到协程完成。
1.4 协程构建器(Coroutine Builders)
  • launch:启动一个协程,不返回结果。
  • async:启动一个协程,并返回一个 Deferred 对象,可用于获取结果。
  • runBlocking:阻塞当前线程,直到协程完成(仅用于测试)。

2. Kotlin 协程的使用方法

以下是协程使用的基本步骤:

2.1 添加依赖项

在项目的 build.gradle 文件中添加 Kotlin 协程库依赖:

dependencies {implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.0'
}
2.2 创建协程作用域

协程必须在协程作用域中运行。例如:

import kotlinx.coroutines.*fun main() = runBlocking { // 用于测试,实际开发中避免使用val scope = CoroutineScope(Dispatchers.Default)scope.launch {// 在这里执行异步任务}
}
2.3 启动协程

使用 launch 或 async 启动协程:

  • launch:用于不需要返回结果的异步任务。
  • async:用于需要返回结果的异步任务。
scope.launch {delay(1000L) // 模拟耗时操作println("异步任务完成")
}val deferred = scope.async {delay(1000L)"异步任务结果"
}
println("异步任务返回值: ${deferred.await()}")
2.4 使用挂起函数

挂起函数允许协程在等待任务时挂起,例如 delay

suspend fun fetchData(): String {delay(1000L) // 模拟网络请求return "Data fetched"
}

3. Kotlin 协程的具体示例

以下通过几个常见场景展示协程的实际用法。

3.1 简单的异步任务

使用 launch 启动一个异步任务:

import kotlinx.coroutines.*fun main() = runBlocking {launch {delay(1000L)println("World!")}println("Hello,")
}输出:
Hello,
World!
3.2 并行任务与结果获取

使用 async 和 await 实现并行任务:

import kotlinx.coroutines.*fun main() = runBlocking {val deferred1 = async {fetchDataFromNetwork1()}val deferred2 = async {fetchDataFromNetwork2()}println("Data1: ${deferred1.await()}")println("Data2: ${deferred2.await()}")
}suspend fun fetchDataFromNetwork1(): String {delay(1000L)return "NetworkData1"
}suspend fun fetchDataFromNetwork2(): String {delay(1000L)return "NetworkData2"
}
3.3 取消协程

使用 cancel 方法取消协程:

import kotlinx.coroutines.*fun main() = runBlocking {val job = launch {delay(5000L)println("Task completed")}delay(1000L)job.cancel()println("Coroutine cancelled")
}

4.实际应用示例

网络请求与数据库操作

// 假设有以下挂起函数
suspend fun fetchUserData(): UserData = withContext(Dispatchers.IO) {// 模拟网络请求delay(1000)UserData("John", 30)
}suspend fun saveToDatabase(user: UserData) = withContext(Dispatchers.IO) {// 模拟数据库操作delay(500)println("User saved: $user")
}data class UserData(val name: String, val age: Int)fun main() = runBlocking {// 顺序执行val user = fetchUserData()saveToDatabase(user)// 并发执行多个网络请求val users = listOf("user1", "user2", "user3")val deferredList = users.map { userId ->async { fetchUserDataForId(userId) }}val results = deferredList.awaitAll()println("All users fetched: $results")
}suspend fun fetchUserDataForId(userId: String): UserData {delay(1000)return UserData("$userId-Name", 20 + userId.last().digitToInt())
}

5. 总结

  • 核心概念:协程、挂起函数、协程作用域和协程构建器是协程编程的基础。
  • 使用方法:通过添加依赖、创建作用域、启动协程和使用挂起函数,可以轻松实现异步编程。
  • 适用场景:协程适用于网络请求、文件 IO、CPU 密集型任务等需要并发处理的场景。

相关文章:

  • 【Linux】初见,基础指令(续)
  • 获取文件夹下所有文件的名称
  • Datawhale PyPOTS时间序列5月第4次笔记
  • Docker run -v 的 rw 和 ro 模式_docker ro
  • shp2pgsql 导入 Shp 到 PostGIS 空间数据库
  • MVDR源码(可直接运行)
  • Jmeter(一) - 环境搭建
  • 小白的进阶之路系列之二----人工智能从初步到精通pytorch中分类神经网络问题详解
  • 3D几何建模引擎3D ACIS Modeler核心功能深度解读
  • 视觉语言模型之困:当否定词成为理解的“盲区”
  • 【AI 大模型】盘古大模型简介 ( 创建空间 | 体验模型 | 部署模型 )
  • AMO——下层RL与上层模仿相结合的自适应运动优化:让人形行走操作(loco-manipulation)兼顾可行性和动力学约束
  • ⭐️白嫖的阿里云认证⭐️ 第二弹【课时3:大模型辅助内容生产场景】for 「大模型Clouder认证:利用大模型提升内容生产能力」
  • 第3天-python流程控制实例
  • 保证数据库 + redis在读写分离场景中事务的一致性
  • 隐形安全感
  • 1.3 C++之变量与数据类型
  • 【算法-栈】深入栈模拟题:从题型特征到实现技巧
  • Https流式输出一次输出一大段,一卡一卡的-解决方案
  • Spark离线数据处理实例
  • 外交部:国际调解院着眼以调解定分止争,更好维护国际公平正义
  • 西安市长安区与航天基地区政合一管理,党政一把手分任基地党工委正副书记
  • 在“三只手”上跳舞:公共政策的科学与艺术——读《市场、国家和民众:公共政策经济学》
  • 习近平:推进中国式现代化要继续把制造业搞好
  • 安徽凤阳通报鼓楼瓦片脱落:去年3月维修竣工,已成立调查组
  • 减负举措如何助力基层干部轻装上阵?记者一线调查