一、如何使用协程
1.1 添加依赖
implementation
'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.0.0'
implementation
'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.0.0'
1.2 使用协程Coroutine
在kotlinx.coroutines包中,你可以使用launch或async启动一个协程。从概念上讲,async就像launch一样,它启动一个单独的协程,协程相当于一个轻量级的线程,与其他所有的协同程序同时工作。
async和launch不同的地方在于,launch返回一个Job并且不携带任何结果值,而async返回Deffered。
Deffered表示一个轻量级的非阻塞未来,表示稍后提供结果的承诺。我们可以使用await()方法获取一个deffered的返回结果。Deffered本质上也是Job,因此可以在需要的时候取消它。
如果在launch中的代码因为异常而终止,那么它会被是为线程中未捕获异常而导致应用崩溃。异步代码中未捕获异常存储在生成的Deffered中,并且不会在其他任何地方传递,除非经过处理,否则它会被静默删除。
协程分发
在Android中,我们常用的又两个分发器dispatcher:
uiDispatcher:将执行分发到Android主UI线程(用于父协程)
bgDispatcher:在后台线程中调度执行(用于子协程)
// dispatches execution into Android main thread
val uiDispatcher: CoroutineDispatcher = Dispatcher.Main
// represent a pool of shared thread as coroutine dispatcher
val bgDispatcher: CoroutineDispatcher = Dispatcher.IO
协程作用域
使用协程需要提供协程对应的作用域CoroutineScope或使用GlobalScope
// GlobalScope示例
class MainFragment : Fragment(){
fun loadData() = GlobalScope.launch{...}
}
//CoroutineScope示例
class MainFragment : Fragment(){
val uiScope = CoroutineScope(Dispatchers.Main)
fun loadData() = uiScope.launch{...}
}
//Fragment实现CoroutineScope示例
class MainFragment : Fragment(),CoroutineScope{
override val coroutineContext: CoroutineContext
get() = Dispatcher.Main
fun loadData() = launch {...}
}
lauch+async(执行任务)
父协程通过Main Dispatcher调用的launch方法启动。
子协程通过IO Dispatcher调用async方法启动。
Note:父协程会一直等待它的子协程完成
Note:协程如果发生未捕获异常,程序会崩溃
val uiScope = CoroutineScope(Dispatchers.Main)
fun loadData() = uiScope.launch {
view.showLoading() //ui thread
val task = async(bgDispatcher){ //background thread
// your blocking call
}