前言
大家好,我是小益!在经过前两章对协程的介绍后,我们终于又回到了 MVVM的封装 。协程在Android开发中最常用的场景应该是网络请求了,其次是一些使用 Thread 的场景,本章内容我们将着重介绍如何将协程与网络请求结合。
一、viewModelScope的使用
自行创建协程
var uiScope = CoroutineScope(SupervisorJob() + Dispatchers.Main)
在上述代码中我们创建了一个协程并指定了这个协程是在主线程中工作,之后我们就可以使用前两章提到的 launch 来操作了,如下:
uiScope.launch{
...
}
以上是我们创建协程的实现方式,我们可以通过指定 Dispatchers 来决定协程到底在什么线程中工作,而其实 Kotlin 的协程核心库中也为我们提供封装好了的 scope ,例如 MainScope ,源码如下:
@Suppress("FunctionName")
public fun MainScope(): CoroutineScope = ContextScope(SupervisorJob() + Dispatchers.Main)
非常明显, Kotlin 提供的 MainScope 内部实现与我们自行创建的 CoroutineScope 一模一样, MainScope 在一定程度上方便了我们创建协程。
lifecycle-viewmodel-ktx
知晓协程如何创建后,我们需要思考一个问题:协程主要的使用层是 MVVM 的哪一层?因为协程最主要的作用是用同步编码的方式来实现异步;既然有异步,那么直接操作UI的View层明显是不太适合使用协程的,剩下的ViewModel与Model层则都很适合添加协程封装。我们先从ViewModel开始添加协程,幸运的是Google已经考虑到了这一层,并为我们提供了相关依赖,导入方式如下:
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.2.0'
在导入此依赖后,会为 ViewModel 添加一个名为 viewModelScope 的扩展函数,此函数会创建一个做了优化的协程,源码如下:
val ViewModel.viewModelScope: CoroutineScope
get() {
val scope: CoroutineScope? = this.getTag(JOB_KEY)
if (scope != null) {
return scope
}
return setTagIfAbsent(JOB_KEY,
CloseableCoroutineScope(SupervisorJob() + Dispatchers.Main.immediate))
}
拿到 viewModelScope 后,我们就可以在 BaseViewModel 添加如下代码:
abstract class BaseViewModel : ViewModel(), ViewModelLifecycle, ViewBehavior {
/**
* 在主线程中执行一个协程
*/
protected fun launchOnUI(block: suspend CoroutineScope.() -> Unit): Job {
return viewModelScope.launch(Dispatchers.Main) { block() }
}
/**
* 在IO线程中执行一个协程
*/
protected fun launchOnIO(block: suspend CoroutineScope.() -> Unit): Job {
return viewModelScope.launch(Dispatchers.IO) { block() }
}
}
二、与Retrofit的结合
目前在Android开发中,最主流的网络请求框架应该就是 Retrofit+OkHttp+RxJava 这一套了。那么下面我们就使用 Retrofit 来结合协程进行封装。在网络请求中,协程起的作用其实与 RxJava 是一致的,所以如果在别处没有使用 RxJ