协程是一种并发设计模式,您可以在 Android 平台上使用它来简化异步执行的代码。协程是在版本 1.3 中添加到 Kotlin 的,它基于来自其他语言的既定概念。

在 Android 上,协程有助于管理长时间运行的任务,如果管理不当,这些任务可能会阻塞主线程并导致应用无响应。使用协程的专业开发者中有超过 50% 的人反映使用协程提高了工作效率。本主题介绍如何使用 Kotlin 协程解决以下问题,从而让您能够编写出更清晰、更简洁的应用代码。

特点

协程是我们在 Android 上进行异步编程的推荐解决方案。值得关注的特点包括:

轻量:您可以在单个线程上运行多个协程,因为协程支持挂起,不会使正在运行协程的线程阻塞。挂起比阻塞节省内存,且支持多个并行操作。

内存泄漏更少:使用结构化并发机制在一个作用域内执行多项操作。

内置取消支持:取消操作会自动在运行中的整个协程层次结构内传播。

Jetpack 集成:许多 Jetpack 库都包含提供全面协程支持的扩展。某些库还提供自己的协程作用域,可供您用于结构化并发。

示例概览

根据应用架构指南,本主题中的示例会发出网络请求并将结果返回到主线程,然后应用可以在主线程上向用户显示结果。

具体而言,ViewModel 架构组件会在主线程上调用代码库层,以触发网络请求。本指南介绍了多种使用协程确保主线程畅通的解决方案。

ViewModel 包含一组可直接与协程配合使用的 KTX 扩展。这些扩展是 lifecycle-viewmodel-ktx 库,在本指南中有用到。

依赖项信息

如需在 Android 项目中使用协程,请将以下依赖项添加到应用的 build.gradle 文件中:

dependencies {

implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9'

}

在后台线程中执行

如果在主线程上发出网络请求,则主线程会处于等待或阻塞状态,直到收到响应。由于线程处于阻塞状态,因此操作系统无法调用 onDraw(),这会导致应用冻结,并有可能导致弹出“应用无响应”(ANR) 对话框。为了提供更好的用户体验,我们在后台线程上执行此操作。

首先,我们来了解一下 Repository 类,看看它是如何发出网络请求的:

sealed class Result {

data class Success(val data: T) : Result()

data class Error(val exception: Exception) : Result()

}

class LoginRepository(private val responseParser: LoginResponseParser) {

private const val loginUrl = "https://example.com/login"

// Function that makes the network request, blocking the current thread

fun makeLoginRequest(

jsonBody: String

): Result {

val url = URL(loginUrl)

(url.openConnection() as? HttpURLConnection)?.run {

requestMethod = "POST"

setRequestProperty("Content-Type", "application/json; utf-8")

setRequestProperty("Accept", "application/json")

doOutput = true

outputStream.write(jsonBody.toByteArray())

return Result.Success(responseParser.parse(inputStream))

}

return Result.Error(Exception("Cannot open HttpURLConnection"))

}

}

makeLoginRequest 是同步的,并且会阻塞发起调用的线程。为了对网络请求的响应建模,我们创建了自己的 Result 类。

ViewModel 会在用户点击(例如,点击按钮)时触发网络请求:

class LoginViewModel(

private val loginRepository: LoginRepository

): ViewModel() {

fun login(username: String, token: String) {

val jsonBody = "{ username: \"$username\", token: \"$token\"}"

loginRepository.makeLoginRequest(jsonBody)

}

}

使用上述代码,LoginViewModel 会在网络请求发出时阻塞界面线程。如需将执行操作移出主线程,最简单的方法是创建一个新的协程,然后在 I/O 线程上执行网络请求:

class LoginViewModel(

private val loginRepository: LoginRepository

): ViewModel() {

fun login(username: String, token: String) {

// Create a new coroutine to move the execution off the UI thread

viewModelScope.launch(Dispatchers.IO) {

val jsonBody = "{ username: \"$username\", token: \"$token\"}"

loginRepository.makeLoginRequest(jsonBody)

}

}

}

下面我们仔细分析一下 login 函数中的协程代码:

viewModelScope 是预定义的 CoroutineScope,包含在 ViewModel KTX 扩展中。请注意,所有协程都必须在一个作用域内运行。一个 CoroutineScope 管理一个或多个相关的协程。

launch 是一个函数,用于创建协程并将其函数主体的执行分派给相应的调度程序。

Dispatchers.IO 指示此协程应在为 I/O 操作预留的线程上执行。

login 函数按以下方式执行:

应用从主线程上的 View 层调用 login 函数。

launch 会创建一个新的协程,并且网络请求在为 I/O 操作预留的线程上独立发出。

在该协程运行时,login 函数会继续执行,并可能在网络请求完成前返回。请注意,为简单起见,我们暂时忽略掉网络响应。

由于此协程通过 viewModelScope 启动,因此在 ViewModel 的作用域内执行。如果 ViewModel 因用户离开屏幕而被销毁,则 viewModelScope 会自动取消,且所有运行的协程也会被取消。

前面的示例存在的一个问题是,调用 makeLoginRequest 的任何项都需要记得将执行操作显式移出主线程。下面我们来看看如何修改 Repository 以解决这一问题。

使用协程确保主线程安全

如果函数不会在主线程上阻止界面更新,我们即将其视为是主线程安全的。makeLoginRequest 函数不是主线程安全的,因为从主线程调用 makeLoginRequest 确实会阻塞界面。可以使用协程库中的 withContext() 函数将协程的执行操作移至其他线程:

class LoginRepository(...) {

...

suspend fun makeLoginRequest(

jsonBody: String

): Result {

// Move the execution of the coroutine to the I/O dispatcher

return withContext(Dispatchers.IO) {

// Blocking network request code

}

}

}

withContext(Dispatchers.IO) 将协程的执行操作移至一个 I/O 线程,这样一来,我们的调用函数便是主线程安全的,并且支持根据需要更新界面。

makeLoginRequest 还会用 suspend 关键字进行标记。Kotlin 利用此关键字强制从协程内调用函数。

注意:为更轻松地进行测试,我们建议将 Dispatchers 注入 Repository 层。如需了解详情,请参阅在 Android 上测试协程。

在以下示例中,协程是在 LoginViewModel 中创建的。由于 makeLoginRequest 将执行操作移出主线程,login 函数中的协程现在可以在主线程中执行:

class LoginViewModel(

private val loginRepository: LoginRepository

): ViewModel() {

fun login(username: String, token: String) {

// Create a new coroutine on the UI thread

viewModelScope.launch {

val jsonBody = "{ username: \"$username\", token: \"$token\"}"

// Make the network call and suspend execution until it finishes

val result = loginRepository.makeLoginRequest(jsonBody)

// Display result of the network request to the user

when (result) {

is Result.Success -> // Happy path

else -> // Show error in UI

}

}

}

}

请注意,此处仍需要协程,因为 makeLoginRequest 是一个 suspend 函数,而所有 suspend 函数都必须在协程中执行。

此代码与前面的 login 示例的不同之处体现在以下几个方面:

launch 不接受 Dispatchers.IO 参数。如果您未将 Dispatcher 传递至 launch,则从 viewModelScope 启动的所有协程都会在主线程中运行。

系统现在会处理网络请求的结果,以显示成功或失败界面。

login 函数现在按以下方式执行:

应用从主线程上的 View 层调用 login() 函数。

launch 创建一个新的协程,以在主线程上发出网络请求,然后该协程开始执行。

在协程内,调用 loginRepository.makeLoginRequest() 现在会挂起协程的进一步执行操作,直至 makeLoginRequest() 中的 withContext 块结束运行。

withContext 块结束运行后,login() 中的协程在主线程上恢复执行操作,并返回网络请求的结果。

注意:如需与 ViewModel 层中的 View 通信,请按照应用架构指南中的建议,使用 LiveData。遵循此模式时,ViewModel 中的代码会在主线程上执行,因此您可以直接调用 MutableLiveData 的 setValue() 函数。

处理异常

为了处理 Repository 层可能抛出的异常,请使用 Kotlin 对异常的内置支持。在以下示例中,我们使用的是 try-catch 块:

class LoginViewModel(

private val loginRepository: LoginRepository

): ViewModel() {

fun makeLoginRequest(username: String, token: String) {

viewModelScope.launch {

val jsonBody = "{ username: \"$username\", token: \"$token\"}"

val result = try {

loginRepository.makeLoginRequest(jsonBody)

} catch(e: Exception) {

Result.Error(Exception("Network request failed"))

}

when (result) {

is Result.Success -> // Happy path

else -> // Show error in UI

}

}

}

}

在此示例中,makeLoginRequest() 调用抛出的任何意外异常都会处理为界面错误。

其他协程资源

如需详细了解 Android 上的协程,请参阅利用 Kotlin 协程提升应用性能。

如需获取更多协程资源,请访问以下链接:

android 协程,Android 上的 Kotlin 协程相关推荐

  1. pdf 深入理解kotlin协程_深入理解Kotlin协程

    领取成功 您已领取成功! 您可以进入Android/iOS/Kindle平台的多看阅读客户端,刷新个人中心的已购列表,即可下载图书,享受精品阅读时光啦! - | 回复不要太快哦~ 回复内容不能为空哦 ...

  2. Android - kotlin 协程极简入门

    背景 协程的作用是解决并发,并发的常见场景是多线程. "并发"和"并行"经常弄混.下面是我的理解,当然我理解也不一定是对的 "并发"是宏观上 ...

  3. 枯燥的Kotlin协程三部曲(上)——概念启蒙篇

    0x0.引言 Kotlin 1.3 版本开始引入协程 Coroutine,简练的官方文档和网上一堆浅尝辄止的文章让我心里有些没底,不想止步于仅仅知道: ① Android中,Kotlin协程用于解决: ...

  4. 一文看透 Kotlin 协程本质

    前言 公司开启新项目了,想着准备亮一手 Kotlin 协程应用到项目中去,之前有对 Kotlin 协程的知识进行一定量的学习,以为自己理解协程了,结果--实在拿不出手! 为了更好的加深记忆和理解,更全 ...

  5. Kotlin 协程,怎么开始的又是怎么结束的?原理讲解!

    九心 | 作者 承香墨影 | 校对 https://juejin.cn/post/6862548590092140558 | 原文 Hi,大家好,这里是承香墨影! 上周我们聊到 Kotlin 协程的使 ...

  6. Kotlin 协程探索

    文章目录 Kotlin 协程是什么? suspend 是什么? 总结 Kotlin 协程是什么? 本文只是自己经过研究后,对 Kotlin 协程的理解概括,如有偏差,还请斧正. 简要概括: 协程是 K ...

  7. 探索 Kotlin 协程原理

    接下来跟大家分享一下我在了解 Kotlin 协程实现的过程中理解的一些概念,如果你发现哪些地方我说错了的话,欢迎提出你的理解. 1. Kotlin 协程原理概述 Kotlin 协程的大致的执行流程如上 ...

  8. 在 Android 开发中使用 Kotlin 协程 (一) -- 初识 Kotlin 协程

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

  9. Android Kotlin 协程async

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/118093646 本文出自[赵彦军的博客] 往期精彩文章 Android Corou ...

最新文章

  1. 【转】Visual Studio团队资源管理器 Git 源码管理工具简单入门
  2. DOM Node Element Attr 的联系与区别汇总
  3. Design Pattern - Bridge(C#)
  4. Java的Map遍历
  5. php获取当天的开始时间和结束时间
  6. 2020第十一届蓝桥杯软件类省赛第二场C/C++ 大学 B 组(题解)
  7. spinlock剖析与改进
  8. Number.parseInt(), Number.parseFloat()
  9. C#十进制与十六进制转换
  10. Futter基础第19篇: 实现调用第三方时间选择器、日期选择器、时间戳
  11. chrome浏览器插件--让你的谷歌浏览器舒适度提升1800%的插件(程序员推荐)
  12. 3d建模做一单多少钱?做外包赚钱吗?
  13. gpuz怎么看显存颗粒
  14. 【分子结构】原子结构总结——2015年9月12日
  15. mysql int 可以是负数吗_int型包括负数吗
  16. 怎样计算机翼升力大小,如何计算升力和阻力?
  17. 解决vscode中文乱码
  18. python去除视频马赛克_DeepMosaics
  19. 汽车变速器与分动器的讲解(组图)
  20. 【特征提取】基于深度学习的特征提取和匹配方法介绍

热门文章

  1. yamlip 安装_prometheus.(1).yaml安装
  2. cpuz检测硬件真假_一手硬件买不起 淘二手硬件的你知道这些重灾区吗
  3. 什么叫matlab仿真,【图片】求助帖:哪位matlab大神能告诉我这个仿真这能得出什么结论呢_matlab吧_百度贴吧...
  4. python装饰器系列(五)
  5. 农信银高莉:农信科技共享计划
  6. ASP.NET的学习之asp.net整体运行机制
  7. C#基础系列——Attribute特性使用
  8. java基础-关键字-native
  9. 数据绑定 单个对象和集合绑定差异
  10. Jexus vs IIS8 非绝对客观对比测试