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

线程 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

  • 12
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值