Android 上的 Kotlin 协程

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

在 Android 上,协程有助于管理可能阻塞主线程并导致应用无响应的长时间运行的任务。超过 50% 使用协程的专业开发人员报告称生产力有所提高。本主题介绍如何使用 Kotlin 协程解决这些问题,使您能够编写更简洁、更简洁的应用代码。

功能

协程是我们推荐的 Android 异步编程解决方案。值得注意的功能包括:

  • 轻量级:由于支持挂起,因此您可以在单个线程上运行多个协程,而不会阻塞协程运行的线程。挂起在支持许多并发操作的同时节省了比阻塞更多的内存。
  • 更少的内存泄漏:使用结构化并发在范围内运行操作。
  • 内置取消支持取消会自动通过正在运行的协程层次结构传播。
  • Jetpack 集成:许多 Jetpack 库包含提供完整协程支持的扩展。某些库还提供自己的协程作用域,您可以将其用于结构化并发。

示例概述

根据应用架构指南,本主题中的示例进行网络请求并将结果返回到主线程,应用可以在其中将结果显示给用户。

具体来说,ViewModel 架构组件在主线程上调用存储库层以触发网络请求。本指南迭代了使用协程保持主线程不被阻塞的各种解决方案。

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

依赖信息

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

Groovy

dependencies {
    implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9'
}

Kotlin

dependencies {
    implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9")
}

在后台线程中执行

在主线程上进行网络请求会导致它等待或阻塞,直到收到响应。由于线程被阻塞,因此操作系统无法调用onDraw(),这会导致您的应用冻结并可能导致出现“应用程序无响应 (ANR)”对话框。为了获得更好的用户体验,让我们在后台线程上运行此操作。

首先,让我们看一下我们的Repository 类,看看它如何进行网络请求

sealed class Result<out R> {
    data class Success<out T>(val data: T) : Result<T>()
    data class Error(val exception: Exception) : Result<Nothing>()
}

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<LoginResponse> {
        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在进行网络请求时会阻塞 UI 线程。将执行移出主线程的最简单解决方案是创建一个新的协程并在 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是使用ViewModel KTX 扩展包含的预定义CoroutineScope。请注意,所有协程都必须在作用域中运行。CoroutineScope管理一个或多个相关的协程。
  • launch是一个创建协程并将其实体函数的执行调度到相应调度程序的函数。
  • Dispatchers.IO表示此协程应在为 I/O 操作保留的线程上执行。

login 函数按以下方式执行

  • 应用从主线程上的View层调用login 函数。
  • launch创建一个新的协程,并且网络请求在为 I/O 操作保留的线程上独立进行。
  • 在协程运行时,login 函数继续执行并返回,可能在网络请求完成之前。请注意,为简单起见,目前忽略网络响应。

由于此协程是使用viewModelScope启动的,因此它在ViewModel的作用域内执行。如果由于用户从屏幕导航离开而销毁了ViewModel,则viewModelScope会自动取消,并且所有正在运行的协程也会被取消。

之前示例的一个问题是,任何调用makeLoginRequest的代码都需要记住显式地将执行切换到主线程之外。让我们看看如何修改Repository来为我们解决这个问题。

使用协程确保主线程安全

当函数不会阻塞主线程上的 UI 更新时,我们认为它是主线程安全的。 makeLoginRequest 函数不是主线程安全的,因为从主线程调用 makeLoginRequest 会阻塞 UI。使用协程库中的 withContext() 函数将协程的执行转移到另一个线程

class LoginRepository(...) {
    ...
    suspend fun makeLoginRequest(
        jsonBody: String
    ): Result<LoginResponse> {

        // Move the execution of the coroutine to the I/O dispatcher
        return withContext(Dispatchers.IO) {
            // Blocking network request code
        }
    }
}

withContext(Dispatchers.IO) 将协程的执行转移到 I/O 线程,使我们的调用函数主线程安全,并允许 UI 根据需要更新。

makeLoginRequest 也用 suspend 关键字标记。此关键字是 Kotlin 用于强制在协程内部调用函数的方式。

在以下示例中,协程是在 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<LoginResponse> -> // Happy path
                else -> // Show error in UI
            }
        }
    }
}

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

此代码在几个方面与之前的 login 示例有所不同

  • launch 不会接收 Dispatchers.IO 参数。当您不向 launch 传递 Dispatcher 时,从 viewModelScope 启动的任何协程都将在主线程上运行。
  • 网络请求的结果现在被处理以显示成功或失败的 UI。

登录函数现在按如下方式执行

  • 应用从主线程上的 View 层调用 login() 函数。
  • launch 在主线程上创建一个新的协程,并且协程开始执行。
  • 在协程中,对 loginRepository.makeLoginRequest() 的调用现在会挂起协程的进一步执行,直到 makeLoginRequest() 中的 withContext 块完成运行。
  • withContext 块完成后,login() 中的协程在主线程上恢复执行,并带有网络请求的结果。

处理异常

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

class LoginViewModel(
    private val loginRepository: LoginRepository
): ViewModel() {

    fun login(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<LoginResponse> -> // Happy path
                else -> // Show error in UI
            }
        }
    }
}

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

其他协程资源

要更详细地了解 Android 上的协程,请参阅 使用 Kotlin 协程提高应用性能

有关更多协程资源,请参阅以下链接