Kotlin->Kotlin协程的取消机制

取消整个协程作用域CoroutineScope

kotlin 复制代码
Log.i("yang", "Main Start")
val mScope = CoroutineScope(Dispatchers.IO)

val mJob1 = mScope.launch {
    try {
        Log.i("yang", "Job1 Start")
        delay(1000)
        Log.i("yang", "Job1 End")
    } catch (e: Exception) {
        Log.i("yang", "Job1 Cancelled e = ${e.message}")
    }
}

val mJob2 = mScope.launch {
    try {
        Log.i("yang", "Job2 Start")
        delay(2000)
        Log.i("yang", "Job2 End")
    } catch (e: Exception) {
        Log.i("yang", "Job2 Cancelled e = ${e.message}")
    }
}

Log.e("yang", "Main Continue")
Thread.sleep(500)

mScope.cancel()
Log.e("yang", "Main Cancel")
Log.i("yang", "Main End")

// log
2025-01-06 17:36:52.588  1752-1752  yang                    com.example.myapplication            I  Main Start
2025-01-06 17:36:52.593  1752-1752  yang                    com.example.myapplication            E  Main Continue
2025-01-06 17:36:52.593  1752-2367  yang                    com.example.myapplication            I  Job1 Start
2025-01-06 17:36:52.594  1752-2391  yang                    com.example.myapplication            I  Job2 Start
2025-01-06 17:36:53.094  1752-1752  yang                    com.example.myapplication            E  Main Cancel
2025-01-06 17:36:53.094  1752-1752  yang                    com.example.myapplication            I  Main End
2025-01-06 17:36:53.094  1752-2367  yang                    com.example.myapplication            I  Job1 Cancelled e = Job was cancelled
2025-01-06 17:36:53.094  1752-2407  yang                    com.example.myapplication            I  Job2 Cancelled e = Job was cancelled
  • IO线程创建一个协程作用域CoroutineScope(Dispatchers.IO)依次启动两个协程任务,主线程在500ms后取消整个协程作用域,其中的两个协程任务都会取消
  • 主线程在取消协程作用域mScope.cancel()之后,其中两个JobStart语句会执行,End语句不会执行,因为取消协程作用域时,两个Job都抛出了异常

取消没有返回值的协程任务Job

withContext挂起函数检测取消信号

kotlin 复制代码
Log.i("yang", "Main Start")
val mJob = CoroutineScope(Dispatchers.IO).launch {
    try {
        Log.i("yang", "IO Start")
        Thread.sleep(1000) // 模拟耗时任务
        Log.i("yang", "IO End")
        withContext(Dispatchers.Main) {
            Log.i("yang", "Main withContext Start")
        }
    } catch (e: Exception) {
        Log.i("yang", "Launch Exception = ${e.message}")
    }
}
Log.i("yang", "Main Continue")
Thread.sleep(500)
mJob.cancel()
Log.i("yang", "Main End")

// log
2025-01-06 16:13:46.784 29337-29337 yang                    com.example.myapplication            I  Main Start
2025-01-06 16:13:46.785 29337-29337 yang                    com.example.myapplication            I  Main Continue
2025-01-06 16:13:46.785 29337-29622 yang                    com.example.myapplication            I  IO Start
2025-01-06 16:13:47.285 29337-29337 yang                    com.example.myapplication            I  Main End
2025-01-06 16:13:47.785 29337-29622 yang                    com.example.myapplication            I  IO End
2025-01-06 16:13:47.785 29337-29622 yang                    com.example.myapplication            I  Launch Exception = StandaloneCoroutine was cancelled
  • IO线程创建一个协程作用域CoroutineScope(Dispatchers.IO)并调用launch方法返回一个Job对象
    主线程在500ms后取消这个协程,但是协程体并不能立即感知自己被取消了,只会执行到withContext才会检测取消信号
  • IO线程上的所有log都会执行,withContext挂起函数之后的代码不再执行

delay挂起函数检测取消信号

kotlin 复制代码
Log.i("yang", "Main Start")
val mJob = CoroutineScope(Dispatchers.IO).launch {
    try {
        Log.i("yang", "IO Start")
        delay(1000)
        Log.i("yang", "IO End")
    } catch (e: Exception) {
        Log.i("yang", "Launch Exception = ${e.message}")
    }
}
Log.i("yang", "Main Continue")
Thread.sleep(500)
mJob.cancel()
Log.i("yang", "Main End")

// log
2025-01-06 16:35:38.555 29643-29643 yang                    com.example.myapplication            I  Main Start
2025-01-06 16:35:38.558 29643-29643 yang                    com.example.myapplication            I  Main Continue
2025-01-06 16:35:38.559 29643-30198 yang                    com.example.myapplication            I  IO Start
2025-01-06 16:35:39.059 29643-29643 yang                    com.example.myapplication            I  Main End
2025-01-06 16:35:39.060 29643-30200 yang                    com.example.myapplication            I  Launch Exception = StandaloneCoroutine was cancelled
  • IO线程创建一个协程作用域CoroutineScope(Dispatchers.IO)并调用launch方法返回一个Job对象
    主线程在500ms后取消这个协程,但是协程体并不能立即感知自己被取消了,只会执行到delay才会检测取消信号
  • IO线程上只有IO Start这一句Log会执行,delay挂起函数后的代码不再执行

CoroutineScope.isActive变量检测取消信号

kotlin 复制代码
Log.i("yang", "Main Start")
 var mJob = CoroutineScope(Dispatchers.IO).launch {
     try {
         Log.i("yang", "IO Start")
         Thread.sleep(1000)
         if (!isActive){
             Log.i("yang", "IO is not active")
             return@launch
         }
         Log.i("yang", "IO End")
     } catch (e: Exception) {
         Log.i("yang", "Launch Exception = ${e.message}")
     }
 }
 Log.i("yang", "Main Continue")
 Thread.sleep(500)
 mJob.cancel()
 Log.i("yang", "Main End")
Log.i("yang", "Main Continue")
Thread.sleep(500)
mJob.cancel()
Log.i("yang", "Main End")

// log
2025-01-06 16:48:17.754 30216-30216 yang                    com.example.myapplication            I  Main Start
2025-01-06 16:48:17.757 30216-30216 yang                    com.example.myapplication            I  Main Continue
2025-01-06 16:48:17.757 30216-30670 yang                    com.example.myapplication            I  IO Start
2025-01-06 16:48:18.257 30216-30216 yang                    com.example.myapplication            I  Main End
2025-01-06 16:48:18.758 30216-30670 yang                    com.example.myapplication            I  IO is not active
  • IO线程创建一个协程作用域CoroutineScope(Dispatchers.IO)并调用launch方法返回一个Job对象
    主线程在500ms后取消这个协程,但是协程体并不能立即感知自己被取消了,只会执行到!isActive才会检测取消信号,提前return
  • IO线程上只有IO Start这一句Log会执行,!isActive后的代码不再执行

CoroutineScope.ensureActive()函数检测取消信号

kotlin 复制代码
Log.i("yang", "Main Start")
var mJob = CoroutineScope(Dispatchers.IO).launch {
    try {
        Log.i("yang", "IO Start")
        Thread.sleep(1000)
        ensureActive()
        Log.i("yang", "IO End")
    } catch (e: Exception) {
        Log.i("yang", "Launch Exception = ${e.message}")
    }
}
Log.i("yang", "Main Continue")
Thread.sleep(500)
mJob.cancel()
Log.i("yang", "Main End")

// log
2025-01-06 17:17:36.781 31763-31763 yang                    com.example.myapplication            I  Main Start
2025-01-06 17:17:36.784 31763-31763 yang                    com.example.myapplication            I  Main Continue
2025-01-06 17:17:36.785 31763-32227 yang                    com.example.myapplication            I  IO Start
2025-01-06 17:17:37.285 31763-31763 yang                    com.example.myapplication            I  Main End
2025-01-06 17:17:37.785 31763-32227 yang                    com.example.myapplication            I  Launch Exception = StandaloneCoroutine was cancelled
  • IO线程创建一个协程作用域CoroutineScope(Dispatchers.IO)并调用launch方法返回一个Job对象
    主线程在500ms后取消这个协程,但是协程体并不能立即感知自己被取消了,只会执行到ensureActive()才会检测取消信号
  • IO线程上只有IO Start这一句Log会执行,ensureActive()后的代码不再执行

取消有返回值的协程任务Deferred

  • 取消有返回值的协程任务Deferred可以复用取消没有返回值的协程任务Job的取消方式
  • 取消整个协程作用域,也会同时取消其中启动的有返回值的协程任务Deferred
相关推荐
恋猫de小郭30 分钟前
Flutter Widget IDE 预览新进展,开始推进落地发布
android·前端·flutter
weixin_3077791338 分钟前
PySpark实现ABC_manage_channel逻辑
开发语言·python·spark
lybugproducer2 小时前
创建型设计模式之:简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式和原型模式
java·设计模式·建造者模式·简单工厂模式·工厂方法模式·抽象工厂模式·面向对象
??? Meggie2 小时前
【Python】保持Selenium稳定爬取的方法(防检测策略)
开发语言·python·selenium
南客先生2 小时前
马架构的Netty、MQTT、CoAP面试之旅
java·mqtt·面试·netty·coap
Minyy112 小时前
SpringBoot程序的创建以及特点,配置文件,LogBack记录日志,配置过滤器、拦截器、全局异常
xml·java·spring boot·后端·spring·mybatis·logback
百锦再2 小时前
Java与Kotlin在Android开发中的全面对比分析
android·java·google·kotlin·app·效率·趋势
武昌库里写JAVA3 小时前
39.剖析无处不在的数据结构
java·vue.js·spring boot·课程设计·宠物管理
酷爱码3 小时前
如何通过python连接hive,并对里面的表进行增删改查操作
开发语言·hive·python
画个大饼4 小时前
Go语言实战:快速搭建完整的用户认证系统
开发语言·后端·golang