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
来改写:
coroutineScope.launch(Dispatchers.Main) {
tv_name.text = “xxx”
// 获取用户信息
val userInfo = withContext(Dispatchers.Main) {
getUserInfo()
}
tv_name.text = userInfo.username
// 获取消息列表
val msgList = withContext(Dispatchers.IO) {
getMessageList(userInfo!!.token)
}
// 显示消息列表
}
在从上述代码中,withContext
都是顺序执行,不像多个同级的launch
是并发执行。使用withContext
可以消除launch
的嵌套,阅读体验更佳。不过withContext
只能在CoroutineScope
内部使用,不能单独在其他方法中使用,比如:
class BaseActivity : Activity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
withContext(Dispatchers.IO){
…
}
}
}
这种写法是不行的,因为withContext
只能在CoroutineScope
内部使用。
五、小结
本章内容简单介绍了协程是什么以及协程的基本使用,下章内容将会介绍写协程中的async/await
和suspend
关键字,这两者在我们的日常开发中使用频率非常高,尤其是suspend
。最后推荐一个Kotlin协程认知系列文章,强烈推荐对协程不是很理解的同学去阅读!
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
最后附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题 (含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总)
面试成功其实是必然的,因为我做足了充分的准备工作,包括刷题啊,看一些Android核心的知识点,看一些面试的博客吸取大家面试的一些经验,下面这份PDF是我翻阅了差不多1个月左右一些Android大博主的博客从他们那里取其精华去其糟泊所整理出来的一些Android的核心知识点, 全部都是精华中的精华,我能面试到现在资深开发人员跟我整理的这本Android核心知识点有密不可分的关系,在这里本着共赢的心态分享给各位朋友。
这份PDF囊括了JVM,Java集合,Java多线程并发,Java基础,生命周期,微服务, 进程,Parcelable 接口,IPC,屏幕适配,线程异步,ART,架构,Jetpack,NDK开发,计算机网络基础,类加载器,Android 开源库源码分析,设计模式汇总,Gradle 知识点汇总…
由于篇幅有限,就不做过多的介绍,大家请自行脑补
058)]
这份PDF囊括了JVM,Java集合,Java多线程并发,Java基础,生命周期,微服务, 进程,Parcelable 接口,IPC,屏幕适配,线程异步,ART,架构,Jetpack,NDK开发,计算机网络基础,类加载器,Android 开源库源码分析,设计模式汇总,Gradle 知识点汇总…
由于篇幅有限,就不做过多的介绍,大家请自行脑补