kotlin 携程

1. CoroutineScope

CoroutineScope 是协程运行的范围。每个协程都需要一个 CoroutineScope,以便管理其生命周期。

kotlin 复制代码
CoroutineScope(Dispatchers.Default).launch {
    // 在此范围内启动一个协程
}

2. Job

Job 是一个协程的句柄,它表示协程的生命周期,并允许显式地取消协程。

kotlin 复制代码
val job = CoroutineScope(Dispatchers.Default).launch {
    // 协程体
}
// 取消协程
job.cancel()

3. Dispatcher

Dispatcher 决定了协程在哪个线程池上执行。常用的 Dispatcher 有:

  • Dispatchers.Main - 用于在主线程上执行协程。
  • Dispatchers.IO - 用于执行 I/O 操作。
  • Dispatchers.Default - 用于 CPU 密集型任务。
kotlin 复制代码
CoroutineScope(Dispatchers.IO).launch {
    // 在 I/O 线程上运行
}

4. async 和 await

async 用于并发地执行多个任务,并返回一个 Deferred 对象,可以使用 await 来获取结果。

kotlin 复制代码
val deferred = CoroutineScope(Dispatchers.Default).async {
    // 异步计算
    42
}
val result = deferred.await()  
/Suspend function 'await' should be called only from a coroutine or another suspend function

5. launch

launch 启动一个新的协程,不返回任何结果。

kotlin 复制代码
CoroutineScope(Dispatchers.Main).launch {
    // 在主线程上运行的协程
}

6. withContext

withContext 用于在指定的 Dispatcher 上执行代码块,并返回结果。

kotlin 复制代码
val result = withContext(Dispatchers.IO) {
    // 在 I/O 线程上执行
    42
}

7. SupervisorJob

SupervisorJob 是一种特殊类型的 Job,它的子协程失败不会取消父协程。

kotlin 复制代码
val scope = CoroutineScope(SupervisorJob() + Dispatchers.Main)
scope.launch {
    // 子协程1
}
scope.launch {
    // 子协程2
}

8. Exception Handling

协程中的异常处理可以使用 try-catchCoroutineExceptionHandler

kotlin 复制代码
val handler = CoroutineExceptionHandler { _, exception ->
    println("Caught $exception")
}
CoroutineScope(Dispatchers.Main + handler).launch {
    try {
        // 协程体
    } catch (e: Exception) {
        // 处理异常
    }
}
9.例子
import kotlinx.coroutines.*

fun main() = runBlocking {
    var result: Int = 0
    val deferred = CoroutineScope(Dispatchers.Default).async {
        delay(1000)
        42
    }
    result = deferred.await()
    println(result)
}

Kotlin 的挂起函数(suspend function)用于实现异步编程。挂起函数可以在不阻塞线程的情况下挂起执行,并在稍后恢复。挂起函数使用 suspend 关键字定义。它们只能在协程或另一个挂起函数内调用。

kotlin 复制代码
suspend fun fetchData(): String {
    // 模拟网络请求
    delay(1000L)
    return "Data from server"
}

挂起函数的特点

  1. 非阻塞: 挂起函数不会阻塞线程,可以在等待过程中让出线程资源。
  2. 只能在协程中调用: 由于挂起函数可能会挂起当前的执行,需要在协程或另一个挂起函数中调用。

挂起函数的使用

要使用挂起函数,需要在协程上下文中启动它。常用的协程构建器有 launchasync

kotlin 复制代码
import kotlinx.coroutines.*

fun main() = runBlocking {
    launch {
        val result = fetchData()
        println(result)
    }
}

协程上下文与调度器

协程运行在特定的上下文中,调度器 (Dispatcher) 决定了协程运行的线程或线程池。

  • Dispatchers.Main:在主线程上运行,常用于更新 UI。
  • Dispatchers.IO:适合 IO 操作(如网络或文件操作)。
  • Dispatchers.Default:适合 CPU 密集型任务。
kotlin 复制代码
fun main() = runBlocking {
    launch(Dispatchers.IO) {
        val result = fetchData()
        withContext(Dispatchers.Main) {
            // 更新 UI
            println(result)
        }
    }
}

挂起函数组合

可以组合多个挂起函数,实现复杂的异步操作。

kotlin 复制代码
suspend fun fetchUserData(): String {
    // 模拟网络请求
    delay(1000L)
    return "User data"
}

suspend fun fetchAdditionalData(): String {
    // 模拟网络请求
    delay(1000L)
    return "Additional data"
}

fun main() = runBlocking {
    launch {
        val userData = fetchUserData()
        val additionalData = fetchAdditionalData()
        println("User Data: $userData, Additional Data: $additionalData")
    }
}

异常处理

协程中的异常处理可以使用 try-catch 块。

kotlin 复制代码
fun main() = runBlocking {
    launch {
        try {
            val result = fetchData()
            println(result)
        } catch (e: Exception) {
            println("Error: ${e.message}")
        }
    }
}
相关推荐
奋斗的小花生15 分钟前
c++ 多态性
开发语言·c++
魔道不误砍柴功17 分钟前
Java 中如何巧妙应用 Function 让方法复用性更强
java·开发语言·python
闲晨20 分钟前
C++ 继承:代码传承的魔法棒,开启奇幻编程之旅
java·c语言·开发语言·c++·经验分享
老猿讲编程1 小时前
一个例子来说明Ada语言的实时性支持
开发语言·ada
Chrikk2 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*2 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue2 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man2 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang
萧鼎3 小时前
Python并发编程库:Asyncio的异步编程实战
开发语言·数据库·python·异步
学地理的小胖砸3 小时前
【一些关于Python的信息和帮助】
开发语言·python