0

0

C++如何实现协程_C++20 coroutine基础语法与co_await/co_yield入门

穿越時空

穿越時空

发布时间:2025-12-01 16:29:02

|

737人浏览过

|

来源于php中文网

原创

C++20协程通过co_await、co_yield、co_return实现暂停与恢复,需返回类型含promise_type以控制行为,如用co_yield构建生成器range(from, to)逐次产出值。

c++如何实现协程_c++20 coroutine基础语法与co_await/co_yield入门

C++20 引入了原生的协程支持,让异步编程变得更加直观。协程是一种可以暂停和恢复执行的函数,它不同于普通函数,不会一次性运行到底,而是在某些点上“挂起”,之后可以从挂起点继续执行。

协程的基本概念

在 C++20 中,协程通过三个关键词标识:

  • co_await:用于等待一个可等待对象(awaiter),执行到此处时可能挂起协程。
  • co_yield:将一个值“产出”并挂起协程,常用于生成器场景。
  • co_return:结束协程,并可返回结果或通知完成。

只要函数体内包含上述任意一个关键字,该函数就被视为协程,编译器会自动生成相应的状态机代码来管理其挂起与恢复逻辑。

协程返回类型要求

要使一个函数成为合法协程,其返回类型必须满足特定条件,即定义了 promise_type 内嵌类型。这个 promise_type 负责控制协程的行为,比如初始化、返回值处理、异常处理和最终清理。

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

例如,我们定义一个简单的生成器返回类型:

struct Generator {
    struct promise_type {
        int current_value;
    // 协程开始前调用
    std::suspend_always initial_suspend() { return {}; }
    // 协程结束后是否挂起
    std::suspend_always final_suspend() noexcept { return {}; }
    // 设置返回值
    Generator get_return_object() { return Generator{this}; }
    // co_return 时调用
    void return_void() {}
    // 处理未捕获异常
    void unhandled_exception() { std::terminate(); }

    // 支持 co_yield value
    std::suspend_always yield_value(int value) {
        current_value = value;
        return {};
    }
};

// 其他成员:构造、迭代接口等略

private: promise_type* p; };

使用 co_yield 实现生成器

利用上面定义的 Generator 类型,我们可以写一个能逐步产生数值的协程:

Generator range(int from, int to) {
    for (int i = from; i < to; ++i) {
        co_yield i; // 暂停并返回当前值
    }
}

每次调用 co_yield 时,协程会保存当前状态并挂起,直到被外部恢复。这非常适合实现惰性序列,如范围迭代器、数据流等。

HeyGen
HeyGen

HeyGen是一个AI虚拟数字人生成平台,可以根据用户提供的内容,快速生成高质量的虚拟发言人视频,支持数字化身、文本转视频和视频翻译。

下载

使用 co_await 等待异步操作

co_await 用于等待一个“可等待”(awaitable)对象。如果该对象表示的操作已完成,则不挂起;否则协程暂停,直到被唤醒。

简单示例:等待一个总是立即完成的对象:

struct ImmediateAwaiter {
    bool await_ready() { return true; }  // 立即完成
    void await_suspend(std::coroutine_handle<>) {}
    int await_resume() { return 42; }
};

Generator example() { int val = co_await ImmediateAwaiter{}; co_yield val; // 输出 42 }

更复杂的 awaiter 可以注册回调,在 I/O 完成后恢复协程句柄,实现真正的异步非阻塞行为。

协程句柄与手动控制

每个协程都有一个对应的 std::coroutine_handle,可用于手动控制其生命周期:

  • handle.resume():恢复挂起的协程。
  • handle.done():检查协程是否已完成。
  • handle.destroy():销毁协程资源。

通过 handle,你可以从外部驱动协程执行,适用于事件循环或任务调度系统。

基本上就这些。C++20 协程虽然语法简洁,但底层机制复杂,理解 promise_type 和 awaiter 的交互是掌握它的关键。实际项目中建议结合已有的库(如 cppcoro)来简化开发。

相关专题

更多
C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

3

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

7

2026.01.15

windows查看wifi密码教程大全
windows查看wifi密码教程大全

本专题整合了windows查看wifi密码教程大全,阅读专题下面的文章了解更多详细内容。

30

2026.01.15

浏览器缓存清理方法汇总
浏览器缓存清理方法汇总

本专题整合了浏览器缓存清理教程汇总,阅读专题下面的文章了解更多详细内容。

2

2026.01.15

ps图片相关教程汇总
ps图片相关教程汇总

本专题整合了ps图片设置相关教程合集,阅读专题下面的文章了解更多详细内容。

8

2026.01.15

ppt一键生成相关合集
ppt一键生成相关合集

本专题整合了ppt一键生成相关教程汇总,阅读专题下面的的文章了解更多详细内容。

3

2026.01.15

php图片上传教程汇总
php图片上传教程汇总

本专题整合了php图片上传相关教程,阅读专题下面的文章了解更多详细教程。

2

2026.01.15

phpstorm相关教程大全
phpstorm相关教程大全

本专题整合了phpstorm相关教程汇总,阅读专题下面的文章了解更多详细内容。

4

2026.01.15

热门下载

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

精品课程

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

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.4万人学习

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

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