协程的启动和取消

1、协程构建器

launch与async构建器都用来启动新协程

launch,返回一个Job并且不附带任何结果值

async,返回一个Deferred,Deferred也是一个Job,可以使用await()在一个延期的值上得到它的最终结果

等待一个作业

launch通过join

async通过await

2、协程的启动模式

DEFAULT:协程创建后,立即开始调度,在调度前如果协程被取消,其将直接进入取消响应的状态

ATOMIC:协程创建后,立即开始调度,协程执行到第一个挂起点之前不响应取消

LAZY:只有协程被需要时,包括主动调用协程的start、join或await等函数时才会开始调度,如果调度前就被取消,那么该协程直接进入异常结束状态

UNDISPATCHED:协程创建后立即在当前函数调用栈中执行,直到遇到第一个真正挂起的点

3、协程的作用域构建器

coroutionScope与runBlocking

runBlocking是常规函数,而coroutionScope是挂起函数

它们都会等待其协程体以及所有子协程结束,主要区别在于runBlocking方法会阻塞当前线程来等待,而coroutionScope只是挂起,会释放底层线程用于其他用途

coroutionScope与supervisorScope

coroutionScope:一个协程失败了,所有其它兄弟协程也会被取消

supervisorScope:一个线程失败了,不会影响其他兄弟协程

4、结构化并发

使用结构化并发可以做到:

取消任务,当某任务不再需要时取消它

追踪任务,当任务正在执行时,追踪它

发出错微信号,当携程失败时,发出错误信号表明有错误发生

5、Job对象

对于每一个创建的协程(通过launch或者async),会返回一个Job实例,该实例是协程的唯一标识,并且负责管理协程的生命周期

一个任务可以包含一系列的状态:新创建(New)、活跃(Active)、完成中(Completing)、已完成(Completed)、取消中(Cancelling)和已取消(Cancelled)

我们无法直接访问这些状态,但是我们可以访问Job的属性:isActive、isCancelled和isCompleted

6、Job的生命周期

 

7、协程的取消

取消作用域会取消它的子协程

被取消的子协程并不会影响其余兄弟协程

协程通过抛出一个特殊的异常CancellationException来处理取消操作

所有kotlin.coroutines中的挂起函数(withContext、delay等)都是可取消的

8、CPU密集型任务取消

isActive是一个可以被使用在CoroutineScope中的扩展属性,检查Job是否处于活跃状态

ensureActive(),如果job处于非活跃状态,这个方法会立即抛出异常

yield函数会检查所在的协程的状态,如果已经取消,则抛出CancellationException予以响应。此外,它还会尝试让出线程的执行权,给其它协程提供执行机会

9、协程取消的副作用

在finally中释放资源

use函数:该函数只能被实现了Closeable的对象使用,程序结束的时候会调用close方法,适合文件对象

10、不能取消的任务

出去取消中状态的协程不能够挂起(运行不能取消订单代码),当协程被取消后需要调用挂起函数,我们需要将清理任务的代码放置于NonCancellable CoroutineContext中

这样会挂起运行中的代码,并保持协程的取消中状态直到任务处理完成

11、超时任务

很多情况下取消一个协程的理由是它有可能超时

withTimeOrNull通过返回null进行超时操作,从而替代抛出一个异常

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
协程中,可以使用协程作用域中的 `cancel()` 函数来取消一个协程。当协程取消时,它会抛出一个 `CancellationException` 异常。如果在协程中捕获了这个异常,那么协程的执行会立即结束。 然而,在某些情况下,我们可能不想让协程取消,或者不想让它在取消时抛出异常。这时候我们就可以使用 `SupervisorJob` 和 `SupervisorScope` 来管理协程取消操作。 `SupervisorJob` 是一个特殊的 `Job` 类型,它会将所有子协程放入一个单独的协程作用域中,从而使它们互相独立。当一个子协程取消时,其它子协程不会受到影响。同时,如果使用 `SupervisorScope` 来启动协程,则在取消时也不会抛出 `CancellationException` 异常。 下面是一个例子,展示了如何在协程中使用 `SupervisorJob` 和 `SupervisorScope`: ```kotlin import kotlinx.coroutines.* fun main() = runBlocking { val supervisor = SupervisorJob() val scope = CoroutineScope(coroutineContext + supervisor) val job1 = scope.launch { println("Child coroutine 1 started") delay(1000) println("Child coroutine 1 completed") } val job2 = scope.launch { println("Child coroutine 2 started") try { delay(3000) println("Child coroutine 2 completed") } catch (e: CancellationException) { // do nothing } } delay(2000) job1.cancel() job2.cancel() supervisor.children.forEach { it.join() } println("All coroutines completed") } ``` 在这个例子中,我们首先创建了一个 `SupervisorJob`,然后使用它来创建一个协程作用域。在协程作用域中,我们创建了两个子协程 `job1` 和 `job2`。在 `job1` 中,我们没有使用 `SupervisorScope`,所以在取消时会抛出 `CancellationException` 异常。而在 `job2` 中,我们使用了 `SupervisorScope`,所以在取消时不会抛出异常。 最后,我们通过遍历 `supervisor.children` 来等待所有子协程执行完毕,并输出 "All coroutines completed"。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值