android kotlin 协程(五) suspend与continuation
android kotlin 协程(五) suspend与continuation
通过本篇你将学会:
-
suspendCoroutine{}
-
suspendCancellableCoroutine{}
-
suspend 与 continuation
suspendCoroutine
第一次看到这玩意的时候肯定有点身体不适, 先不用管这个东西是什么,
目前为止 只需要知道 suspendCoroutine是一个函数即可
先来想想如果不用这个suspendCoroutine ,遇到一个网络请求的原始写法是怎么样的
通常情况下,我们请求一个接口,至少需要处理2种情况
- 成功返回
- 失败返回
来看例子:
private suspend fun requestLoginNetworkData(account: String, pwd: String) =withContext(Dispatchers.IO) {delay(2000)// 模拟请求耗时if (account == "123456789" && pwd == "666666") {Result.success("登陆成功")} else {Result.failure(Throwable("登陆失败"))}}fun main() = runBlocking<Unit> {val deferred = async {// 模拟网络请求requestLoginNetworkData("987654321", "666666")}// 获取网络返回数据,判断成功与失败val result = deferred.await()// result.getOrDefault("") // 如果返回错误使用 默认值// result.getOrThrow() // 如果返回错误使用 错误// result.getOrNull() // 如果返回错误使用 nullresult.onSuccess {println("登陆成功:${result.getOrNull()}")}.onFailure {println("登陆失败:${result.getOrNull()}")}
}
在这段代码中,我们模拟网络请求, 给一个错误的帐号密码,最终打印结果为
登陆失败:null
通过前几篇的了解,这个例子应该是非常简单的
来看看使用 suspendCoroutine怎么玩
private suspend fun <T> requestLoginNetworkData(account: String, pwd: String): String {return withContext(Dispatchers.IO) {delay(2000) // 模拟网络耗时需要2sreturn@withContext suspendCoroutine {if (account == "123456789" && pwd == "666666") {it.resume("登陆成功")} else {it.resumeWithException(RuntimeException("登陆失败"))}}}
}suspend fun main() = runBlocking {val scope = CoroutineScope(Dispatchers.IO)// 开启一个协程val deferred = scope.async {// 模拟网络请求requestLoginNetworkData<String>("987654321", "666666")}// 获取网络返回数据,判断成功与失败val result = runCatching {deferred.await()}if (result.isSuccess) {printlnThread("登陆成功:${result.getOrNull()}")} else {printlnThread("登陆失败:${result.exceptionOrNull()}")}
}
好像使用suspendCoroutine 之后代码变得更多了?
来看看两段代码的区别:
这两段代码,只不过是回调方式不同!
那么是否可以理解为 suspendCoroutine 本质就是一个回调呢?
没错! 暂时可以理解为:suspendCoroutine 就是一个回调
再来看看 suspendCoroutine的具体实现
其本质就是一个Continuation
tips: Continuation 这个角色特别重要,Continuation 是用来使挂起函数恢复执行状态的
就是传说中: kotlin挂起于恢复中的 恢复
再来看看调用的方法:
- resume 恢复正确
- resumeWithException 恢复错误
目前不理解恢复没关系, 先理解为就是一个接口回调
- resume 回调正确
- resumeWithException 回调错误
suspendCoroutine
的本质作用是创建一个挂起点,它会将当前协程挂起,并将协程的执行权交给调用方函数。同时,它会传入一个 Continuation
对象,该对象包含了协程的上下文和协程恢复后需要执行的操作。调用方函数可以在执行完必要的操作后,调用该 Continuation
对象的 resume
方法,来唤醒协程并继续执行。
suspendCoroutine 是一个非常重要的函数,它可以让我们将异步操作转化为同步代码风格
说的直白一点就是:
-
不使用 suspendCoroutine 执行一个suspend的函数的时候, 恢复工作由系统完成
-
使用 suspendCoroutine会将系统的恢复工作抢过来,可以通过 continuation#resume() 来自己恢复
例如这样,我们手动处理了 suspendCoroutine,但是没有恢复, 就会无限挂起
我们知道在kotlin中有suspend,但是在java中并没有suspend关键字,
那么kotlin suspend函数反编译成java后是什么样的
可以看出,suspend关键字并没有任何作用, 他的唯一作用就是告诉开发者,我这里需要挂起罢了
真实干活的其实是 Continuation!
现在你还觉得 suspendCoroutine 仅仅只是一个回调嘛?
suspendCoroutine 不仅可以控制suspend函数的恢复,而且还可以让异步的代码同步化.
最关键的是线程, 线程安全不用我们担心.
来比较一下同步代码与异步代码的风格写法:
也没说异步写法不好,黑猫白猫,抓住老鼠就是好猫,但是这只是一个请求,如果说 逻辑很多,嵌套很深的话,代码会不会成这样:
suspendCancellableCoroutine
suspendCoroutine 与 suspendCancellableCoroutine 的区别:
suspendCancellableCoroutine 相当于是对 suspendCoroutine 的一次封装, 增加了一些 状态,以及 可以 cancel了
- isActive 是否活跃
- isCancelled 是否取消
- isCompleted 是否执行完成
还记得这三个状态吗? Job中也有这三个状态!
suspendCancellableCoroutine 增加了 invokeOnCancellation , 该方法用来监听协程取消, 当协程被取消的时候会被回调
来看看下面的例子:
可以看到,invokeOnCancellation 并没有执行,这里也很好理解,因为没有cancel不执行也正常
在换一个例子
这里的关键点是await, 这是官方的一个扩展,来看看:
这段代码对Cell扩展了一下, 请求数据的时,
- 请求成功 就恢复
- 请求失败 也恢复,只不过会throw异常
当协程取消的时候,将okhttp cancel掉
invokeOnCancellation 注意事项
在使用suspendCancellableCoroutine的时候,有一个方法 invokeOnCancellation
这个方法用来监听当前作用域是否取消
先来看看运行的3种状态:
- isActive 是否活跃
- isCancelled 是否取消
- isCompleted 是否执行完成
当我们调用 Continuation#resume()
恢复之后, 当前协程就会被标记为完成状态
这里有一个小细节:Continuation#cancel()
只能cancel
未完成或在进行中的协程, 如果协程一旦执行完成,也就是一旦恢复,那么 invokeOnCancellation
则不会被调用
再来看看取消:
这种情况,应该大家看看就会了很好理解
还有一种写法, 我们知道,当我们cancel父协程的时候,所有子协程也会被cancel,那么我们就可以利用这个特性,来完成这个效果
例如这样:
这里有一个很关键的点,折磨了我很久:)
当一个挂起函数中的suspendCancellableCoroutine
函数被恢复(例如,通过调用continuation.resume
或continuation.resumeWithException
)后,该协程就不再挂起,并且不能再被取消。因此,在恢复之后,该协程将无法响应invokeOnCancellation
函数。
完整代码
下篇开始会看看协程源码, 以及手动创建协程等
下篇预告:
- SafeContinuation
- startCoroutine
- createCoroutine
- receiver startCoroutine
- receiver createCoroutine
原创不易,您的点赞就是对我最大的支持!
相关文章:

android kotlin 协程(五) suspend与continuation
android kotlin 协程(五) suspend与continuation 通过本篇你将学会: suspendCoroutine{} suspendCancellableCoroutine{} suspend 与 continuation suspendCoroutine 第一次看到这玩意的时候肯定有点身体不适, 先不用管这个东西是什么, 目前为止 只需要知道 suspendCoro…...

JavaScript事件循环
大厂面试题分享 面试题库后端面试题库 (面试必备) 推荐:★★★★★地址:前端面试题库一、异步执行原理1. 单线程的JavaScript我们知道,JavaScript是一种单线程语言,它主要用来与用户互动,以及操…...

华为OD机试真题Python实现【最少停车数】真题+解题思路+代码(20222023)
最少停车数 题目 特定大小的停车场 数组cars表示 其中1表示有车0表示没车 车辆大小不一,小车占一个车位(长度1) 货车占两个车位(长度2) 卡车占三个车位(长度3) 统计停车场最少可以停多少辆车 返回具体的数目 🔥🔥🔥🔥🔥👉👉👉👉👉👉 华为OD机试(Pyt…...

Python每日一练(20230223)
目录 1. 合并区间 2. 单词接龙 3. N皇后 附录:回溯算法 基本思想 一般步骤 1. 合并区间 难度:★★ 以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间,并返回…...

Flask----------第一个flask项目,debug、host、port的配置
目录 1.flask 1.简介 2.flask框架的优势 2.第一个flask项目 3.debug 开启debug方法 1.专业版 2.社区版 4.修改host 5. 修改port端口 1.flask 1.简介 Flask是一个基于Python开发并且依赖jinja2模板和Werkzeug WSGI服务的一个微型框架,对于Werkzeug本质是So…...

容器技术概述
容器技术概述 软件应用程序通常依赖于运行时环境提供的其他库、配置文件或服务。软件应用程序的传统运行环境是物理主机或虚拟机,应用程序依赖项作为主机的一部分安装。 例如,考虑一个 Python 应用程序,它需要访问实现 TLS 协议的公共共享库…...

「SAP」ABAP模块学习需要了解什么?快收下这份ABAP技术栈指南【附技能树】
💂作者简介: THUNDER王,一名热爱财税和SAP ABAP编程以及热爱分享的博主。目前于江西师范大学会计专业大二本科在读,阿里云社区专家博主,华为云社区云享专家,CSDN SAP应用技术领域新兴创作者。 在学习工…...

【python 基础篇 九】python的常用数据类型操作-------时间日历
目录1.python时间操作1.1 time模块1.2 calendar模块1.3 datetime模块1.python时间操作 python程序能用很多方式处理日期和时间,转换日期格式也是一个常见功能。 1.1 time模块 提供了处理时间和表示之间转换的功能 获取当前时间戳 概念:从0时区的1…...

华为OD机试真题Python实现【相同字符连续出现的最大次数】真题+解题思路+代码(20222023)
相同字符连续出现的最大次数 题目 输入一串字符串 字符串长度不超过100 查找字符串中相同字符连续出现的最大次数 🔥🔥🔥🔥🔥👉👉👉👉👉👉 华为OD机试(Python)真题目录汇总 ## 输入 输入只有一行,包含一个长度不超过100的字符串 输出描述 输出只…...

【Unity3D】空间和变换
1 空间 1.1 左右手坐标系及其法则 1.1.1 左右手坐标系 左手坐标系与右手坐标系Unity 局部空间、世界空间、裁剪空间、屏幕空间都采用左手坐标系,只有观察空间采用右手坐标系。 左右手坐标系除了坐标系朝向(旋向性)不同,还存在以…...

脑洞|ChatGPT加持下,ChatOps将如何革新团队协作与运维管理?
要说近期科技圈 “顶流”,非 ChatGPT 莫属。 比起目前常见的语音助手与聊天 bot,这位机器人显得更有 “人味儿”,不仅能模拟人类的语气,跟你聊得有来有回,还能写剧本、编音乐、写代码。 说到聊天工具,就让…...

华为OD机试真题Python实现【找数字】真题+解题思路+代码(20222023)
找数字 题目 给一个二维数组nums,对于每一个元素num[i],找出距离最近的且值相等的元素,输出横纵坐标差值的绝对值之和,如果没有等值元素,则输出-1。 例如: 输入数组nums为 0 3 5 4 2 2 5 7 8 3 2 5 4 2 4对于 num[0][0] = 0,不存在相等的值。 对于 num[0][1] = 3,存…...

【Database-01】达梦数据库Docker版下载安装
1、前往达梦数据库官网下载 https://www.dameng.com/1.1、选择数据库 - 数据库产品系 1.2、选择 达梦数据库管理系统(DM8) 1.3、点击试用下载 1.4、注册达梦账户 1.5、选择DM8 Docker镜像 https://www.dameng.com/list_103.html1.6、或者使用以下网址也…...

Allegro如何打开格点显示效果操作指导
Allegro如何打开格点显示效果操作指导 Allegro可以设置格点显示效果,以格点来判定走线等等是否都处于格点上,如下图 如何打开格点显示效果,具体操作如下 点击Setup点击Grids...

电子技术——反馈放大器的分析方法总结
电子技术——反馈放大器的分析方法总结 第一种也是最简单的估算方法,直接拿出反馈网络,计算 β\betaβ 则假设在 AβA\betaAβ 无限大的情况下有 Af≃1/βA_f \simeq 1/\betaAf≃1/β 。开环法。比第一种方法更能精确的估计 AAA 和 β\betaβ 的值。系…...

微服务系统启动,环境从0开始的搭建过程
1. JDK的下载安装(傻瓜式) 安装过程傻瓜式,直接一步到位。我安装的版本为:jdk-17_windows-x64_bin 2. 集成开发工具的下载安装:IDEA(傻瓜式) ideaIU-2021.2.1 网上资源很多,自己找…...

手工测试1年经验面试,张口要13K,我真是服了····
由于朋友临时有事, 所以今天我代替朋友进行一次面试,他需要应聘一个测试工程师, 我以很认真负责的态度完成这个过程, 大概近30分钟。 主要是技术面试, 在近30分钟内, 我与被面试者是以交流学习的方式进行的…...

【保姆级】手把手捋动态代理流程(JDK+Cglib超详细源码分析)
简介动态代理,通俗点说就是:无需声明式的创建java代理类,而是在运行过程中生成"虚拟"的代理类,被ClassLoader加载。 从而避免了静态代理那样需要声明大量的代理类。上面的简介中提到了两个关键的名词:“静态…...

Appium自动化测试 Inspector定位Webview/H5页面元素
目录操作步骤Python操作该混合App代码Appium在操作混合App或Android App的H5页面时, 常常需要定位H5页面中的元素, 传统方式是 FQ 使用Chrome://inspect来定位元素, 环境准备相当繁琐, 不仅需要想办法FQ, 而且还需要Android设备安装Google框架以及手机版Chrome浏览器以及相应的…...

数组求和方法总结,学点干货
1.循环 (新手用) 1.1 普通for 循环 简单质朴 const arr [1, 2, 3, 4, 5];let sum 0;for (let i 0; i < arr.length; i) {sum arr[i];}1.2 for in 循环 与普通for循环大同小异 const arr [1, 2, 3, 4, 5];let sum 0;for (let i in arr) {sum …...

斗地主洗牌发牌-课后程序(JAVA基础案例教程-黑马程序员编著-第六章-课后作业)
【案例6-4】 斗地主洗牌发牌 【案例介绍】 1.任务描述 扑克牌游戏“斗地主”,相信许多人都会玩,本案例要求编写一个斗地主的洗牌发牌程序,要求按照斗地主的规则完成洗牌发牌的过程。一副扑克总共有54张牌,牌面由花色和数字组成…...

基于antd封装的二次业务筛选组件-table-filter
文档地址:https://flowerofsummer.github.io/components/ 业务筛选组件 支持各种类型的高级搜索组件 基础用法 组件响应式布局,默认显示两行,可以通过 maxLineCount 配置最多显示行数每行个数: 如果含有 time-range࿰…...

逆向-还原代码之max 再画堆栈图 (Interl 64)
// source code #include <stdio.h> void max(int * a, int * b) { if (*a < *b) *a *b; } int main() { int a 5, b 6; max(&a, &b); printf("a, b max %d\n", a); return 0; } // 再画堆栈图 下周一(2.27…...

GitHub标星30K+的Java面试八股文长啥样?
2023年的互联网行业竞争越来越严峻,面试也是越来越难,一直以来我都想整理一套完美的面试宝典,奈何难抽出时间,这套1000道的Java面试手册我整理了整整1个月,上传到Git上目前star数达到了30K 一、32 道 MySQL 面试题 1&…...

CVE-2022-39197 POC(CobaltStrike XSS <=4.7)漏洞复现
漏洞说明 根据9.20日CobaltStrike官方发布的最新4.7.1版本的更新日志中介绍,<4.7的teamserver版本存在XSS漏洞,从而可以造成RCE远程代码执行 一位名为“Beichendream”的独立研究人员联系我们,告知我们他们在团队服务器中发现的一个 XSS …...

我们来说说蹿红的AIGC到底是什么?ChatGPT又是什么?
近期,人工智能(AI)领域动作频频,OPENAI公司Chat GPT的出现,标志着人工智能的研究与应用已经进入了一个崭新的发展阶段,国内腾讯、阿里巴巴、百度、易网、国外微软、谷歌、苹果、IBM、Amazon,等互…...

新手如何从零开始搭建配置Windows云服务器?
新手如何从零开始搭建配置Windows云服务器?本文是搭建 Windows 云服务器入门教程,主要介绍如何从零开始,以最简单的方式搭建和配置你的Windows 云服务器。如果您之前没有搭建云服务器的经验,建议您按照本文介绍的方式来购买和配置…...

百趣代谢组学-抑郁症居然“男女有别”,脑膜淋巴管起关键作用!
文章标题:A functional role of meningeal lymphatics in sex difference of stress susceptibility in mice 发表期刊:Nature Communications 影响因子:17.694 发表时间:2022年8月 作者单位:中山大学中山医学院 …...

C语言实现用堆解决 TOP-K 问题
目录 TopK函数实现 如何测试 完整源码 生活中我们经常能见到TopK问题,例如:专业前10名、世界500强、富豪榜、游戏中前100的活跃玩家等。 所以,TopK问题即求出一组数据中前K个最大或最小的元素,一般情况下,数据量都…...

MySQL 数据库基础命令
MySQL 基础命令 一.了解数据库 1、了解数据库对象 1.表: 用于以有组织方式存储数据。以行和列的格式包含数据。 2.索引: 是内部表结构,MySQL 用它基于一列或多列的值来提供对表中各行的快速访问。 3.视图: 是虚拟表&#…...