全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

c++中的协程是如何挂起和恢复的_c++ coroutine_handle详解【底层】

C++20协程的挂起与恢复由coroutine_handle显式控制,依赖awaitable对象的await_suspend()行为;resume()跳转至挂起点下一条指令,destroy()显式释放帧内存。

在 C++20 中,协程的挂起与恢复并非由语言直接调度,而是通过用户定义的 coroutine_handle 显式控制——它本质是一个轻量级、可复制的指针包装器,指向协程帧(coroutine frame)的内存首地址。真正决定“何时挂起”“如何恢复”的,是协程函数体内 co_await 表达式所作用的 awaitable 对象的行为,而 coroutine_handle 是执行这些行为的唯一入口。

挂起:由 awaitable 的 await_suspend() 触发

当协程执行到 co_await expr 时,编译器会调用 expr.await_suspend(handle)(若存在)。这个函数的返回值决定了挂起后的行为:

  • 返回 void:协程立即挂起,控制权交还给调用者;后续必须手动调用 handle.resume() 才能继续
  • 返回 booltrue 表示已自行安排恢复(如投递到线程池),协程挂起且不自动恢复;false 表示未安排,协程立即恢复(即“不挂起”,相当于同步执行)
  • 返回另一个 coroutine_handle:协程挂起,并将控制权转移给该 handle(常用于链式协程或尾调用优化)

注意:挂起操作本身不释放协程帧内存,只是将协程状态设为 suspended,并保存当前栈上下文(寄存器、局部变量等)到帧中。

coroutine_handle 的构造与有效性

它不能直接构造,只能通过以下方式获得:

  • co_await 表达式中传入的参数(即 await_suspend 的形参)
  • promise_type::get_return_object_on_allocation_failure() 等 promise 回调中(需配合自定义分配器)
  • 从其他 handle 派生:handle.address() 得到原始指针,再用 from_address() 重建(仅当确定内存有效时)

调用 handle.done() 可判断是否已结束(初始/已恢复完毕/已销毁);handle.promise() 返回对 promise 对象的引用,用于读写协程状态;handle.address() 返回帧起始地址,可用于内存管理或调试。

恢复:resume() 的底层含义

handle.resume() 并非“唤醒线程”,而是跳转回协程上次挂起点的下一条指令(由编译器生成的恢复标签决定)。它要求:

  • handle 必须有效且指向一个已挂起(!done())但未被销毁的协程帧
  • 协程帧内存必须仍可访问(未被 operator delete 释放)
  • 恢复时不会重新进入 promise 构造或初始 suspend,而是从中断处继续执行

若在 resume 前已调用 destroy(),或帧内存被复用,行为是未定义的。因此,典型模式是:在 await_suspend 返回 void 后,由外部事件(IO 完成、定时器触发、另一线程通知)安全地调用 resume()

销毁协程帧:显式 vs 隐式

协程帧的生命周期独立于 handle:

  • handle.destroy():手动释放帧内存(调用 promise 的析构、operator delete),之后 handle 变为无效
  • 隐式销毁:当协程执行到末尾(或抛出未捕获异常),且处于 final_suspend 状态时,编译器会自动调用 destroy() —— 这正是为什么 final_suspend 的 awaitable 通常返回 std::suspend_always(挂起等待显式销毁)或自定义逻辑(如自动回收)

常见错误是 resume 后未及时 destroy,导致内存泄漏;或过早 destroy 后仍尝试 resume,引发崩溃。正确做法是:final_suspend 返回的 awaitable 应明确管理帧生命周期。

基本上就这些。协程没有运行时调度器,没有隐式上下文切换,一切挂起/恢复/销毁都经由 coroutine_handle 和 awaitable 协同完成——它不是语法糖,而是一套可预测、可调试、完全暴露给用户的协作式控制流原语。


#   # ai  # c++  # 为什么  # 局部变量  # bool  # void  # 指针  # operator  # 线程  # 形参  # delete  # 对象  # 事件  # promise  # 挂起  # 自定义  # 跳转  # 链式  # 隐式  # 未被  # 是一个  # 设为  # 并将  # 下一条 


相关文章: 网站网页制作电话怎么打,怎样安装和使用钉钉软件免费打电话?  香港网站服务器数量如何影响SEO优化效果?  头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  早安海报制作网站推荐大全,企业早安海报怎么每天更换?  怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?  弹幕视频网站制作教程下载,弹幕视频网站是什么意思?  建站之星安装步骤有哪些常见问题?  如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法  开封网站制作公司,网络用语开封是什么意思?  电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?  企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?  如何零基础在云服务器搭建WordPress站点?  如何获取上海专业网站定制建站电话?  韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐  香港服务器WordPress建站指南:SEO优化与高效部署策略  制作网站外包平台,自动化接单网站有哪些?  如何通过商城免费建站系统源码自定义网站主题?  如何在局域网内绑定自建网站域名?  建站之星后台搭建步骤解析:模板选择与产品管理实操指南  Java解压缩zip - 解压缩多个文件或文件夹实例  如何用搬瓦工VPS快速搭建个人网站?  佛山企业网站制作公司有哪些,沟通100网上服务官网?  建站主机核心功能解析:服务器选择与网站搭建流程指南  Python路径拼接规范_跨平台处理说明【指导】  视频网站制作教程,怎么样制作优酷网的小视频?  建站主机选哪家性价比最高?  如何在阿里云通过域名搭建网站?  专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?  免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  如何在腾讯云服务器快速搭建个人网站?  兔展官网 在线制作,怎样制作微信请帖?  制作销售网站教学视频,销售网站有哪些?  如何在Ubuntu系统下快速搭建WordPress个人网站?  建站之星备案流程有哪些注意事项?  如何正确选择百度移动适配建站域名?  香港服务器选型指南:免备案配置与高效建站方案解析  测试制作网站有哪些,测试性取向的权威测试或者网站?  制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?  c++怎么实现高并发下的无锁队列_c++ std::atomic原子变量与CAS操作【详解】  c# 在高并发下使用反射发射(Reflection.Emit)的性能  如何在Golang中使用encoding/gob序列化对象_存储和传输数据  如何在宝塔面板中修改默认建站目录?  存储型VPS适合搭建中小型网站吗?  如何在IIS服务器上快速部署高效网站?  学校免费自助建站系统:智能生成+拖拽设计+多端适配  如何通过建站之星自助学习解决操作问题?  建站之星如何实现PC+手机+微信网站五合一建站?  学校建站服务器如何选型才能满足性能需求?  如何用美橙互联一键搭建多站合一网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。