0

0

c++20的协程(Coroutines)是什么 编写高效的异步代码【详解】

尼克

尼克

发布时间:2025-12-27 05:35:26

|

888人浏览过

|

来源于php中文网

原创

C++20协程是可挂起/恢复的语言级函数机制,非线程亦非普通函数调用;其本质是将执行状态保存至堆上coroutine frame,依赖promise、awaiter和handle三组件,需配合executor实现高效异步调度。

c++20的协程(coroutines)是什么 编写高效的异步代码【详解】

C++20 协程不是线程,也不是简单的函数调用替换,而是一种语言级的、可挂起和恢复的函数机制,用于编写逻辑清晰、资源可控的异步代码。它不依赖操作系统调度,不自动并发,但为 async/await 风格的异步编程提供了底层支撑——真正高效的关键,在于避免回调地狱、减少拷贝、复用内存,并与 executor(执行器)协同调度。

协程的本质:可暂停的函数

普通函数执行完就销毁栈帧;协程函数(用 co_awaitco_yieldco_return 标记)在遇到挂起点时,会把当前执行状态(局部变量、PC 位置等)保存到堆上的一块内存(即 coroutine frame),然后返回控制权。之后可在任意时机恢复执行,从上次暂停处继续。

  • 协程函数返回类型必须满足 std::coroutine_traits 要求,通常自定义一个 TaskGeneratorFuture 类型
  • 编译器自动生成三个关键组件:promise 对象(管理生命周期和返回值)、awaiter(决定何时挂起/恢复)、coroutine handle(用于手动调度或传递)
  • 没有运行时调度器 —— co_await 后是否真“等待”,取决于你提供的 awaiter 的 await_ready()await_suspend() 实现

如何写出高效的异步协程代码

高效 ≠ 写得短,而是减少分配、避免隐式拷贝、明确调度意图、适配执行上下文。

  • 复用协程帧内存:默认 new/delete 分配帧,高频协程开销大。可通过 promise 类型重载 operator new/operator delete 使用对象池或栈内存(如 std::array<char, 1024> + placement new)
  • 避免值拷贝:传参尽量用 const lvalue 引用或 move;返回 large object 前考虑用 co_yield 流式产出,或让 promise 存储指针而非副本
  • co_await 表达等待语义,而非逻辑分支:比如不要写 if (ready()) co_await something();,而是让 awaiter 的 await_ready() 返回 true/false,由编译器决定跳过挂起
  • 绑定执行器(executor):在 await_suspend() 中把 handle 提交给线程池、IO 多路复用器(如 epoll/kqueue 封装)或特定线程,实现真正的异步调度

一个轻量级 awaiter 示例(无栈 IO 等待)

假设你封装了一个基于 epollasync_read 操作:

立即学习C++免费学习笔记(深入)”;

听脑AI
听脑AI

听脑AI语音,一款专注于音视频内容的工作学习助手,为用户提供便捷的音视频内容记录、整理与分析功能。

下载
struct epoll_awaiter {
  https://www.php.cn/link/d3e4b9f9aa7aac3bd930abb82fab2d2b fd_;
  std::span<std::byte> buf_;
  bool ready_ = false;
<p>bool await<em>ready() { return ready</em>; }</p><p>void await_suspend(std::coroutine_handle<> h) {
// 注册 fd 到 epoll,就绪后调用 h.resume()
register_for<em>read(fd</em>, <a href="https://www.php.cn/link/d3e4b9f9aa7aac3bd930abb82fab2d2b">h</a> { h.resume(); });
}</p><p>void await_resume() {} // 无需返回值
};

这样,co_await epoll_awaiter{fd, buf} 就是零拷贝、无栈挂起、事件驱动的等待 —— 不占线程、不阻塞、不轮询。

常见误区与提醒

  • 协程本身不提供并发:多个协程可在同一线程串行调度,也可分发到多线程,取决于你的 executor 设计
  • 不要在协程中裸写 sleep(1)std::this_thread::sleep_for:这会阻塞整个执行器线程。应使用基于 timerfd 或 event loop 的 awaitable sleep
  • 异常传播需显式处理:promise 的 unhandled_exception() 必须定义,否则程序 terminate
  • 移动语义要小心:coroutine handle 是可移动不可复制的,跨 await 边界传递时注意所有权

协程的价值不在语法炫技,而在把异步控制流还原为接近同步的书写体验,同时保留底层掌控力。写得好,它比 callback 更省内存、比 thread 更低开销、比 future 更易组合。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

564

2023.09.20

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

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

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

766

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

377

2025.12.24

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

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

精品课程

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

共48课时 | 10.7万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

TypeScript全面解读课程
TypeScript全面解读课程

共26课时 | 5.1万人学习

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

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