以上是我们创建协程的实现方式,我们可以通过指定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的结合
目前