前言

最近在研究 Kotlin 协程,发现功能真的超级强大,很有用,而且很好学,如果你正在或计划使用 Kotlin 开发 Android,那么 Kotlin 协程你一定不能错过!

协程是什么?

我们平常接触的都是进程、线程,在开发中使用最多的就是线程,比如主线程、子线程,而且操作系统里最小可操作的单元就是线程,那协程又是什么?协程是比线程更小的单位,但并不是说在操作系统里最小可操作单元就从线程变成了协程,而是协程依然运行在线程上,协程是在语言上实现的比线程更小的单位。

那么你可能有疑问,既然协程还是运行在线程上,那直接使用线程不就好了吗?但问题是往往我们用不好线程,首先创建一个线程的成本很高,在 Android 中创建一个线程,大约要消耗 1M 的内存,而且,如果使用线程池,线程间数据的同步也是一个非常麻烦复杂的事情,所以就有了协程:

  • 可以看作是轻量级线程,创建一个协程的成本很低
  • 可以轻松的挂起和恢复操作
  • 支持阻塞线程的协程和不阻塞线程的协程
  • 可以更好的实现异步和并发

如果简单来理解 Kotlin 协程的话,就是封装好的线程池。

Kotlin协程库:Kotlin.coroutines

实现协程的库是 Kotlin.coroutines,点击查看 Kotlin.coroutines 在 Github 上的源码。

Kotlin 是一门支持 多平台的语言,所以 Kotlin.coroutines 也是支持多平台的,包括:

  • Kotlin/JS
  • Kotlin/Native 包括 PC 和 Android

我们使用 Kotlin.coroutines 的 Android 版本。

给 Android 工程添加 Kotlin 协程库

要使用协程,Kotlin 的版本必须在1.3以上。

升级 Kotlin 到 最新版本 1.3.+

在 Android Studio 中选择 Android Stuido -> Preference... -> Languages & Framewroks -> Kotlin

在这里升级 Kotlin

创建使用 Kotlin 开发的 Android 工程

在 Android Studio 中选择 File -> New -> New Project...

在这个界面里选中 Include Kotlin support ,剩下的和创建一般 Android 工程是一样的。

配置 Kotlin 协程库的依赖

app/build.gradle 里添加 Kotlin 协程库的依赖:

implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.1.1'
复制代码

创建 Coroutine(协程):Coroutine Builder

创建Coroutine需要使用 Coroutine Builder 函数,包括:

作用
launch 创建一个不会阻塞当前线程、没有返回结果的 Coroutine,但会返回一个 Job 对象,可以用于控制这个 Coroutine 的执行和取消
runBlocking 创建一个会阻塞当前线程的Coroutine

其实不止这两个,但本篇先介绍这两个。

launch

使用 GlobalScope.launch 来创建协程,使用方法如下:

override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)var job = GlobalScope.launch(Dispatchers.Main) {var content = fetchData()Log.d("Coroutine",content)}
}suspend fun fetchData():String{delay(2000)return "content"
}
复制代码

Activity 的 onCreate() 里,用 GlobalScope.launch 创建一个协程,在协程里我模拟了一个请求,去获取数据,然后把数据打印出来。

因为 GlobalScope.launch无阻塞的,所以不会阻塞 UI 线程。

这里 GlobalScope.launch 创建之后,会返回一个 Job 对象,Job 对象可以这么使用:

  • job.cancel() : 取消协程
  • job.join() :让协程运行完

runBlocking

使用 runBlocking 来创建协程,使用方法如下:

override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)runBlocking {var content = fetchData()Log.d("Coroutine",content)}
}suspend fun fetchData():String{delay(2000)return "content"
}
复制代码

功能和上一个例子一样,但是这里协程创建改成了 runBlocking(),但是 runBlocking() 是会阻塞线程的,所以这里会阻塞 UI 线程,这里是一个错误用例的示范(Orz...)

suspend 方法

在前面介绍协程的代码里,有个不起眼的函数:

suspend fun fetchData():String{delay(2000)return "content"
}
复制代码

suspend 方法是协程里的特有方法。

suspend 方法的定义

suspend 方法的声明很简单,只需在方法 或 Lambda 定义前面加 suspend 关键字即可。

suspend 方法的使用限制

suspend 方法使用由限制,只能有两个地方允许使用 suspend 方法:

  1. 在协程内部使用
  2. 在另一个 suspend 方法里使用

如果你在一个普通方法内存使用 suspend 方法,是会报语法错误的。

suspend 方法的功能

suspend 方法能够使协程执行暂停,等执行完毕后在返回结果,同时不会阻塞线程。

是不是很神奇,只暂停协程,但不阻塞线程。

而且暂停协程里方法的执行,直到方法返回结果,这样也不用写 Callback 来取结果,可以使用同步的方式来写异步代码,真是漂亮啊。

Coroutine context 与 Coroutine dispatchers

想要使用协程,还有两个重要的元素:

  • Coroutine context:协程上下文
  • Coroutine dispatchers :协程调度

Coroutine context:协程上下文

协程上下文里是各种元素的集合。具体的之后的文章在讲。

Coroutine dispatchers :协程调度

我们已经知道协程是运行在线程上的,我们获取数据后要更新 UI ,但是在 Android 里更新 UI 只能在主线程,所以我们要在子线程里获取数据,然后在主线程里更新 UI。这就需要改变协程的运行线程,这就是 Coroutine dispatchers 的功能了。

Coroutine dispatchers 可以指定协程运行在 Android 的哪个线程里。

我们先看下 dispatchers 有哪些种类:

作用
Dispatchers.Default 共享后台线程池里的线程
Dispatchers.Main Android主线程
Dispatchers.IO 共享后台线程池里的线程
Dispatchers.Unconfined 不限制,使用父Coroutine的现场
newSingleThreadContext 使用新的线程

在看前面的代码里,细心的你肯定发现:

var job = GlobalScope.launch(Dispatchers.Main) {var content = fetchData()Log.d("Coroutine",content)}
复制代码

GlobalScope.launch 后面的 Dispatchers.Main 就是指定协程在 Android 主线程里运行。

那么,如何切换线程呢?如下:

GlobalScope.launch(Dispatchers.IO) {...withContext(Dispatchers.Main) {...}
}
复制代码

使用 withContext 切换协程,上面的例子就是先在 IO 线程里执行,然后切换到主线程。

Android 开发中使用 协程

讲完协程的基本用法,你还是不知道改如何用到自己的代码里,这里给出一个最基本的用法,后续的使用方法会不断补充。

首先 MainActivity 要 实现 CoroutineScope 这个接口,CoroutineScope 的实现教由代理类 MainScope,所以是这样子的:

class MainActivity : AppCompatActivity(),CoroutineScope by MainScope()
复制代码

这样 MainActivity 就是一个协程,那么要获取数据,并展示在 UI 上,就可以这么写:

class MainActivity : AppCompatActivity(),CoroutineScope by MainScope() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)setSupportActionBar(toolbar)//加载并显示数据loadDataAndShow()}fun loadDataAndShow(){GlobalScope.launch(Dispatchers.IO) {//IO 线程里拉取数据var result = fetchData() withContext(Dispatchers.Main){//主线程里更新 UItext.setText(result)}}}suspend fun fetchData():String{delay(2000)return "content"}override fun onDestroy() {super.onDestroy()//取消掉所有协程内容cancel()}}
复制代码
  1. 完全不用担心会阻塞主线程
  2. 用同步的方式来写异步代码
  3. 而且不用担心内存泄露的问题

Kotlin 协程,你有没有心动?

未完待续...

转载于:https://juejin.im/post/5c959264f265da60df410c0e

在 Android 开发中使用 Kotlin 协程 (一) -- 初识 Kotlin 协程相关推荐

  1. Android开发中的WMS详细解析

    /   今日科技快讯   / 近日,小冰公司宣布对旗下人工智能数字员工产品线启动年度升级.本次升级加强的技术包括大模型对话引擎.3D神经网络渲染.超级自然语音及AIGC人工智能内容生成.小冰公司计划将 ...

  2. Android开发中应避免的重大错误

    by Varun Barad 由Varun Barad Android开发中应避免的重大错误 (Critical mistakes to avoid in Android development) A ...

  3. 史上最全Android开发中100%会用到的开源框架整理(1/5)

    其实这个开源框架整理很久了,只是一直放在有道云笔记里面,笔者还有很多写得文章都放在有道云笔记里面,有时间都好好整理一下放出来,本篇文章也会不定期更新,由于整理的开源框架分类都有200多个,所有这次只将 ...

  4. android listview下拉动画效果,Android开发中利用ListView实现一个渐变式的下拉刷新动画...

    Android开发中利用ListView实现一个渐变式的下拉刷新动画 发布时间:2020-11-23 16:50:31 来源:亿速云 阅读:80 作者:Leah 本篇文章给大家分享的是有关Androi ...

  5. Android开发中的多线程编程技术

    Android开发中的多线程编程技术 [IT168技术]多线程这个令人生畏的"洪水猛兽",很多人谈起多线程都心存畏惧.在Android开发过程中,多线程真的很难吗?多线程程序的&q ...

  6. android项目中使用的服务器上,android 开发中使用okhttp上传文件到服务器

    开发android手机客户端,常常会需要上传文件到服务器,比如:你手机里的照片. 使用okhttp会是一个很好的选择.它使用很简单,而且运行效率也很高. 首先,在 app/build.gradle 的 ...

  7. Android 开发中的观察者模式应用实例

    前言 最近在遇到了 Android 的开发中常用到的设计模式之观察者模式,观察者模式,所谓的模式就是一种设计思想,可以按照某种模式,写出更合理,简单,有效的代码.可以用在 Android 开发中,也可 ...

  8. Android开发中配置JDK环境的几种方式

    文章目录 一.前言 二.IDE设置 三.环境变量 四.gradle中配置 五.参考链接 一.前言 在Android开发中,配置JDK环境有以下几种方式来是Android项目正常编译 - changin ...

  9. android 4.4 禁止下拉,Android开发中禁止下拉式的实现技巧

    我们开发项目的时候,经常会看到禁止的情况,而Android开发中并没有直接调用的接口,下面是爱站技术频道小编就给大家介绍的Android开发中禁止下拉式的实现技巧,希望网友们喜欢! 分享给大家供大家参 ...

最新文章

  1. 爬取百度百科上中国所有城市的信息
  2. python中setup函数的用法_Vue 3 setup 函数
  3. tomcat装死原因汇总
  4. 苹果cms10自适应模板_哪里有苹果cms10自适应模板?
  5. Vue框架里使用Swiper - 安装篇
  6. 河神,不用砍死那个天秤座的男孩了
  7. php和xml区别,html与xhtml和xml有什么区别
  8. mongodb objectid java_我可以确定字符串是否是MongoDB ObjectID吗?
  9. java volatile关键字的理解
  10. 工作流入门比较经典的文献
  11. koajs mysql_koajs 项目实战(一)
  12. 一句话告诉你为什么有些jQuery插件会有特殊字符(加号、减号、感叹号等)
  13. [渝粤教育] 重庆工商职业学院 生活中的大数据 参考 资料
  14. 计算机体系结构(简记)
  15. 大成都范围广告位招商
  16. COM组件 ATL的创建和调用
  17. APMServ 5.2.6虚拟主机配置bug
  18. 从阿里P1到P7,他的阿里七年总结就是两个字
  19. 美洲杯:阿根廷1-0巴西,梅西破魔咒,阿根廷夺冠
  20. 儿童护眼灯哪个品牌好?精选最好的儿童护眼灯

热门文章

  1. python怎么在电脑上使用-使用python在本地电脑上快速处理数据
  2. python爬虫详细步骤-Python爬虫的步骤和工具
  3. python对英语和数学的帮助-英语和数学都不好,但是我想学Python编程可以吗?
  4. python好学吗mooc中文网-Python语言程序设计
  5. 零基础自学python的app-零基础初学Python,需要装什么软件?
  6. 解决Teamviewer连接出现的黑屏问题
  7. protobuf3在php中的使用
  8. 系统管理员必学的30个Linux实用命令
  9. UVa11968 - In The Airport
  10. Django的下载及命令