0

0

如何在 Kotlin 中将异步回调转换为同步等待的挂起函数

霞舞

霞舞

发布时间:2026-01-11 21:32:02

|

396人浏览过

|

来源于php中文网

原创

如何在 Kotlin 中将异步回调转换为同步等待的挂起函数

本文介绍如何利用 kotlin 华程的 `suspendcancellablecoroutine` 将基于回调的异步 api(如 `areallthesettingsgranted`)安全、简洁地封装为可直接 `await` 的挂起函数,从而避免阻塞线程,实现“逻辑上同步、实际非阻塞”的代码流。

在 Android 或 Kotlin 开发中,许多系统 API(例如 Google Play Services 的 SettingsClient)仍采用回调风格设计,如你提供的 areAllTheSettingsGranted { isGranted -> ... } 和内部调用的 checkLocationSettings(...).addOnSuccessListener {...}。这类 API 本质上是异步的,无法通过 Thread.sleep() 或 CountDownLatch 等方式“强制同步”——那会阻塞主线程、引发 ANR,且违背协程的设计哲学。

✅ 正确解法:使用协程将回调桥接为挂起函数。核心工具是 suspendCancellableCoroutine,它允许你在挂起函数中手动控制协程的恢复时机,并天然支持取消传播与异常处理。

以下是一个完整、生产就绪的封装示例:

import kotlinx.coroutines.suspendCancellableCoroutine
import kotlin.coroutines.resume
import kotlin.coroutines.resumeWithException

// 封装 areAllTheSettingsGranted 为挂起函数
suspend fun Settings.areAllTheSettingsGrantedSuspend(): Boolean {
    return suspendCancellableCoroutine { cont ->
        this.areAllTheSettingsGranted { isGranted ->
            cont.resume(isGranted)
        }
        // 可选:监听协程取消,主动清理资源(如取消 pending 请求)
        cont.invokeOnCancellation {
            // 若 SDK 支持取消回调注册,此处可添加清理逻辑
        }
    }
}

// 同理封装 checkIfLocationServicesAreEnabled
suspend fun Settings.checkIfLocationServicesAreEnabledSuspend(): Boolean {
    return suspendCancellableCoroutine { cont ->
        checkIfLocationServicesAreEnabled { result ->
            cont.resume(result)
        }
    }
}

使用时,只需在协程作用域中调用即可“顺序执行”:

麦艺画板(Max.art)
麦艺画板(Max.art)

AI工业设计平台,专注于汽车设计,线稿、渲染、3D建模全流程覆盖

下载
lifecycleScope.launch {
    try {
        val allGranted = settings.areAllTheSettingsGrantedSuspend()
        if (allGranted) {
            // ✅ 此处 result 已确定,可安全继续后续逻辑
            handlePermissionsGranted()
        } else {
            requestPermissions()
        }
    } catch (e: CancellationException) {
        // 协程被取消(如 Activity 销毁),无需处理
        throw e
    } catch (e: Exception) {
        // 处理其他未预期异常(如回调未触发)
        Log.e("Permission", "Failed to check settings", e)
    }
}

⚠️ 重要注意事项:

  • 切勿在主线程中使用 runBlocking 强制同步:这会冻结 UI,严重违反 Android 最佳实践;
  • 确保调用方处于协程作用域内(如 lifecycleScope、viewModelScope 或自定义 CoroutineScope);
  • suspendCancellableCoroutine 自动集成协程取消机制,若外部协程被取消,cont.invokeOnCancellation 会被触发,应在此处释放相关资源(如注销监听器、关闭流等);
  • 若原始回调可能多次调用(如某些流式 API),需额外加锁或状态检查,防止重复 resume 导致 IllegalStateException;本例中 areAllTheSettingsGranted 是单次回调,故无需防护。

总结:Kotlin 协程不是“让异步变同步”,而是“让异步代码写起来像同步”。通过 suspendCancellableCoroutine,你既保留了非阻塞、响应式的底层能力,又获得了清晰、可读、可组合的线性控制流——这才是现代 Kotlin 异步编程的正确姿势。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

479

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

479

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

60

2025.12.01

android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

267

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1732

2023.08.22

android权限限制怎么解开
android权限限制怎么解开

android权限限制可以使用Root权限、第三方权限管理应用程序、ADB命令和Xposed框架解开。详细介绍:1、Root权限,通过获取Root权限,用户可以解锁所有权限,并对系统进行自定义和修改;2、第三方权限管理应用程序,用户可以轻松地控制和管理应用程序的权限;3、ADB命令,用户可以在设备上执行各种操作,包括解锁权限;4、Xposed框架,用户可以在不修改系统文件的情况下修改应用程序的行为和权限。

1989

2023.09.19

android重启应用的方法有哪些
android重启应用的方法有哪些

android重启应用有通过Intent、PendingIntent、系统服务、Runtime等方法。本专题为大家提供Android相关的文章、下载、课程内容,供大家免费下载体验。

267

2023.10.18

Android语音播放功能实现方法
Android语音播放功能实现方法

实现方法有使用MediaPlayer实现、使用SoundPool实现两种。可以根据具体的需求选择适合的方法进行实现。想了解更多语音播放的相关内容,可以阅读本专题下面的文章。

343

2024.03.01

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 44.8万人学习

Uniapp从零开始实现新闻资讯应用
Uniapp从零开始实现新闻资讯应用

共64课时 | 6.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号