您的位置:首页 > 娱乐 > 八卦 > 1.5-协程基础与关键知识:连接线程的世界-回调型 API 协作

1.5-协程基础与关键知识:连接线程的世界-回调型 API 协作

2024/10/6 14:34:54 来源:https://blog.csdn.net/qq_31339141/article/details/140408935  浏览:    关键词:1.5-协程基础与关键知识:连接线程的世界-回调型 API 协作

文章目录

  • 线程 API 转换成挂起函数:suspendCoroutine
  • 支持取消的 suspendCoroutine:suspendCancellableCoroutine
  • 总结

线程 API 转换成挂起函数:suspendCoroutine

在实际项目中即使已经使用协程了,可是要完全避免跟传统的线程 API 交互并不容易,大型项目一般都会有比较多的老代码或外部库没有用协程,使用的还是回调的写法。那么就很有必要知道怎么让协程和线程 API 的回调交互。

协程有一个专用的函数 suspendCoroutine,它是一个挂起函数,在它里面调用传统的回调式函数,就能把回调式的函数转换成挂起函数

lifecycleScope.launch {val contributors = callbackToSuspend()showContributors(contributors)
}private suspend fun callbackToSuspend() = suspendCoroutine {gitHub.contributorsCall("square", "retrofit").enqueue(object : Callback<List<Contributor>> {override fun onResponse(call: Call<List<Contributor>>, response: Response<List<Contributor>> {// 将结果返回it.resume(response.body()!!)}override fun onFailure(call: Call<List<Contributor>>) {// 发生异常时让 suspendCoroutine 立即结束并抛出这个异常it.resumeWithException(t)}})}
}

使用 suspendCoroutine 包裹的回调式代码需要调用 suspendCoroutine 提供的 continuation.resume 和 continuation.resumeWithException 分别处理正常返回结果和异常的情况

支持取消的 suspendCoroutine:suspendCancellableCoroutine

协程还提供了一个类似的函数 suspendCancellableCoroutine,和 suspendCoroutine 的区别是它支持取消

private suspend fun callbackToCancellableSuspend() = suspendCancellableCoroutine {it.invokeOnCancellation {// 协程被取消,处理协程被取消时要做的一些收尾工作清理现场}gitHub.contributorsCall("square", "retrofit").enqueue(object : Callback<List<Contributor>> {override fun onResponse(call: Call<List<Contributor>>, response: Response<List<Contributor>> {// 将结果返回it.resume(response.body()!!)}override fun onFailure(call: Call<List<Contributor>>) {// 发生异常时让 suspendCoroutine 立即结束并抛出这个异常it.resumeWithException(t)}})}
}

使用 suspendCancellableCoroutine 还可以注册取消的回调,使用 cancellableContinuation.invokeOnCancellation 处理协程被取消时的收尾清理工作

我们用一个例子来说明 suspendCoroutine 和 suspendCancellableCoroutine 的区别:

val job = lifecycleScope.launch {// 假设 callbackToSuspend 会在延时 2s 后继续执行try {val contributors = callbackToSuspend()// val contributors = callbackToCancellableSuspend()showContributors(contributors)} catch (e: Exception) {textView.text = e.message}
}
lifecycleScope.launch {delay(200)job.cancel() // 200ms 后取消协程
}

假设 callbackToSuspend 函数是使用 suspendCoroutine 包起来的回调代码,会在 2s 后返回结果;协程 200ms 后被取消了,但是里面的代码是不配合的,因为协程的取消本身就是一个状态标记,2s 后还是会继续执行代码。

而如果用 suspendCancellableCoroutine 在 200ms 后会正常取消协程,会在 try-catch 抛出 CancellableException 异常,不会在继续执行后续代码。

我们一般在项目中都使用能支持取消的 suspendCancellableCoroutine,除非特殊需求需要启动后协程取消了也得继续执行才用 suspendCoroutine

总结

  • 将线程 API 的回调式代码用 suspendCoroutine 或 suspendCancellableCoroutine 包住,就能实现将回调式代码转换为挂起函数在协程执行,需要调用提供的 continuation.resume 和 continuation.resumeWithException 分别处理正常返回结果和异常的情况

  • suspendCancellableCoroutine 和 suspendCoroutine 的区别是它支持取消和注册协程取消回调;我们一般在项目中都使用能支持取消的 suspendCancellableCoroutine,除非特殊需求需要启动后协程取消了也得继续执行才用 suspendCoroutine

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com