val 结果A = 协程A执行()
val 结果B = 协程B执行()
val 结果C = 协程C执行(结果A, 结果B)
协程D执行(结果C)
PS:此处特别提示一点,上述协程A与B的逻辑看起来是协程A先执行完后再协程B执行,但其实A、B也是并发执行的
从上述流程中不难看出,其写法直接从异步代码写法变成了同步代码写法,逻辑瞬间清晰了很多,少了很多callback,代码的阅读体验也是直线上升。而这都归功于协程可以灵活的在不同线程之间切换,开发者可以明确的控制协程的结束,再也不用被动的等待通知。说到此处,有的同学可能就想到了RxJava,两者都能实现我们想要的效果,不过两者的设计理念并不相同,语法也是天差地别,有兴趣的同学两者可以都了解一下。
综上,我们可以得出一点,协程可以将异步编码简化,用同步的方式写异步,开发者可以灵活的控制协程的执行与结束以及线程的切换。而这也是我们在Android开发中最在意的特性,协程的其他优点我们暂时不必去了解。
二、导入协程
目前高版本的Android Studio添加Kotlin支持时,已经自动添加Kotlin协程支持了,如果不能使用协程,可以添加如下依赖
implementation “org.jetbrains.kotlinx:kotlinx-coroutines-core:1.3.5”
implementation “org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.5”
三、协程的几种使用方式
1. runBlocking
runBlocking {
getUserInfo(userId)
}
此方法一般不推荐使用,因为它会阻塞线程
2. GlobalScope.launch
GlobalScope.launch {
getUserInfo(userId)
}
此种方法需要慎用,虽然它不会阻塞线程,但是它的生命周期与app一致,并且不能取消
3. 自行创建CoroutineScope
// 此处的context是CoroutineContext,和Activity继承的Context不是同一个东西
val coroutineScope = CoroutineScope(context)
coroutineScope.launch {
getUserInfo(userId)
}
此方法比较推荐,使用此方法我们可以灵活的控制协程的生命周期。所以下面我们主要介绍此种方式的协程使用方式,其他两种方式有兴趣的同学可以自行去了解。
四、协程的简单使用
协程目前常用的几个方法有launch
、withContext
以及async/await
,下面我们主要介绍一下launch
与withContext
,async/await
留到下章与suspend
一起讲解。
launch
首先我们先来看下launch
的源码,简单了解一下launch
是什么!
public fun CoroutineScope.launch(
context: CoroutineContext = EmptyCoroutineContext,
start: CoroutineStart = CoroutineStart.DEFAULT,
block: suspend CoroutineScope.() -> Unit
): Job {
val newContext = newCoroutineContext(context)
val coroutine = if (start.isLazy)
LazyStandaloneCoroutine(newContext, block) else
StandaloneCoroutine(newContext, active = true)
coroutine.start(start, coroutine, block)
return coroutine
}
可以看出,launch
是CoroutineScope
的扩展函数(即launch是CoroutineScope的内部函数),并且最后返回了一个新的协程(即创建了新的Coroutine),具体实现我们暂且不管。其使用方法如下:
// 切换到主线程
coroutineScope.launch(Dispatchers.Main) {
…
}
// 切换到IO线程执行
coroutineScope.launch(Dispatchers.IO) {
…
}
// 小例子
coroutineScope.launch(Dispatchers.IO) {
val userInfo = getUserInfo()
}
tv_name = “xxx”
我们可以通过指定Dispatchers
来切换到不同的线程,如果不指定,则协程默认在其所处方法的线程中执行任务。在{ }
中的代码就是协程需要执行的代码块,在最后的小例子中,tv_name = "xxx"
并不会等待getUserInfo()
执行完毕才执行,因为coroutineScope.launch(Dispatchers.IO) { }
的执行并不会阻塞UI线程的执行,coroutineScope.launch(Dispatchers.IO) { }
就相当于是另开了一个线程,就如同new Thread().start()
一样。所以协程与其外部线程的关系我们一定要理清,我们再通过一个具体的例子看下协程内部:
coroutineScope.launch(Dispatchers.Main) {
tv_name.text = “xxx”
// 切换到IO线程
launch(Dispatchers.IO){
// 获取用户信息
val userInfo = getUserInfo()
// 切换到UI线程
launch(Dispatchers.Main){
// 修改用户名显示
tv_name.text = userInfo.username
// 切换到IO线程
launch(Dispatchers.IO) {
// 获取消息列表
val msgList = getMessageList(userInfo.token)
// 切换到UI线程
launch(Dispatchers.Main){
// 显示消息列表
}
}
}
}
}
在上述例子中,我们先在IO线程中获取用户信息,之后在主线程中更新用户姓名,然后在IO线程中获取消息列表,最后在主线程中显示消息列表。整个流程通过launch
来创建在不同线程中工作的子协程完成,并且完全是一种同步编码的体验。另外,如果两个launch
处于同一层次,如下:
coroutineScope.launch(Dispatchers.Main) {
// 切换到IO线程
launch(Dispatchers.IO){
// 获取用户信息
val userInfo = getUserInfo()
}
// 切换到IO线程
launch(Dispatchers.IO){
// 获取首页信息
val userInfo = getHomeInfo()
}
}
那么getUserInfo()
与getHomeInfo()
会并发执行,也就是说这两个launch
是并行的,这种方式适合于那种只管并发,不用返回结果的场景,例如多个网络请求并发,但是相互之间没有关联。
withContext
虽然使用launch
已经使得整个编码方式由异步变为同步,但是多个launch
嵌套并不美观,我们可以优化一下,使用withContext
来改写:
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
结尾
如何才能让我们在面试中对答如流呢?
答案当然是平时在工作或者学习中多提升自身实力的啦,那如何才能正确的学习,有方向的学习呢?为此我整理了一份Android学习资料路线:
这里是一份BAT大厂面试资料专题包:
好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划。来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。
试资料专题包:
[外链图片转存中…(img-r3txjDig-1712073950518)]
好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划。来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。