最近学习了下协程,在这里分享一下我自己的理解。
1、协程是什么?
在定义协程是什么之前,我们应该先知道协程是做什么的,对于项目而言是新增加的东西?还是替换原有的逻辑,从而获得好处?
我们先看看coroutines最基础的用法:
private fun runCoroutines() {
GlobalScope.launch(Dispatchers.Main) {
val data = getData()//获取网络接口数据
val processedData = processData(data)//处理数据
textView.text = processedData//UI显示数据
}
}
private suspend fun getData(): String {
return withContext(Dispatchers.IO) {
"hen_coder"
}
}
private suspend fun processData(data: String): String {
return withContext(Dispatchers.IO) {
data.split("_")//把"hen_coder" 拆成 ["hen","coder"]
.map { it.capitalize() }//把["hen","coder"]改成["Hen","Coder"]
.reduce { acc, s -> acc + s }//把["Hen","Coder"]改成HenCoder
}
}
launch就是协程的启动函数,可以理解为创建一个协程。
然后协程内部有3步操作,分别是从网络接口获取数据;处理数据;UI显示。
根据以前的知识,获取接口数据是耗时操作,是需要异线程处理的;而处理数据,也有可能是耗时操作,我们需要根据情况来决定是否使用异线程来处理;而最后的UI展示数据,是一定要在主线程中处理的。
如果使用retrofit+rxjava来实现的话,就是异线程执行耗时逻辑,然后通过回调方法,来通知UI界面刷新数据。
但是,在协程的代码中,我们看到了,并没有用到新建线程和回调之类的东西,但是代码运行起来是完全没有问题的,那原因就只有一种,就是协程帮我们把这些事情都完成了。
细心的还可以发现,不只是接口请求,甚至一些耗时操作,协程都可以帮你切换到异线程执行,成功后切回主线程。
所以,我们暂时可以定义协程为:处理多任务并发的手段,最大的特点就是可以自动帮助我们切换线程
2、协程怎么用?
首先,导入coroutines
// Coroutines
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.3.2'
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.2'
然后就可以在代码中使用协程了,创建协程就是上面的代码:
GlobalScope.launch(Dispatchers.Main) {
}
这里的代码需要注意的是,GlobalScope一般是调试测试的时候使用,正常项目不用这个,但是这里只是演示基础,不要紧。launch就是启动一个协程,而Dispatchers.Main就是指定协程内的代码,运行在主线程中。
上面的定义我说,协程可以自动切换线程,而launch中的代码又被指定在主线程中运行了,那耗时操作怎么办?怎么切换到其他线程?
private suspend fun getData(): String {
return withContext(Dispatchers.IO) {
"hen_coder"
}
}
这里的代码涉及到2个新东西,第一个是suspend修饰符,另一个是withContext函数。
suspend的作用就是标志方法为挂起函数,被修饰为挂起函数的函数,只能在协程或者其他挂起函数中调用。
withContext函数的作用就是用来切换线程,后面可以看到Dispatchers.IO,就是我切换到IO线程了。
这样,挂机函数就可以切到其他线程来执行了,执行完又回到launch中的主线程中继续执行,达到了自动切换线程效果。
为什么要用协程?跟普通的线程有什么优缺点?
协程简洁了很多,省略了普通线程的回调,使得代码看起来更加好理解。
线程间的切换,我们只需要指定dispatch来指定就可以了。
缺点:性能差一丝丝,因为suspend函数,对于自动寻回切换线程的操作,其实是相对复杂的,但这一点性能的损耗,相比起协程的优点来说,可以忽略。
协程是轻量级的线程?
看情况。
协程不只是给Android使用的,如果从Android的角度来说,kotlin的协程,并不是轻量级的线程。为什么?因为Android是基于JVM的,JVM能识别线程,但识别不了真正的协程。那为什么能用呢?那是因为Android中的协程,最终还是用线程。所以不能说是轻量级的线程。
非Android使用的协程呢,有些的确是的,可以看协程的官方文档。