0

0

C++怎么使用C++20的协程(Coroutine)_C++异步编程模型与co_await详解

尼克

尼克

发布时间:2025-11-24 13:34:02

|

556人浏览过

|

来源于php中文网

原创

C++20引入协程支持,通过co_await、co_yield和co_return实现异步编程与生成器模式。协程为无栈协程,由编译器管理状态,需定义promise_type控制行为。co_await用于挂起等待异步操作完成,自定义awaitable类型需实现await_ready、await_suspend和await_resume。示例展示延迟任务与整数生成器,协程可简化异步流程,避免回调地狱,适用于网络请求、定时任务等场景。生成器模式利用co_yield按需产生值,节省内存。尽管需手动处理promise与生命周期,协程仍显著提升异步代码可读性与效率。

c++怎么使用c++20的协程(coroutine)_c++异步编程模型与co_await详解

C++20引入了原生的协程支持,让异步编程变得更加直观和高效。协程不是简单的函数调用,而是一种可以暂停和恢复执行的函数。通过co_awaitco_yieldco_return关键字,C++20实现了对异步操作、生成器等模式的原生支持。下面详细介绍如何使用C++20协程进行异步编程。

协程基本概念与语法

C++20协程是无协程(stackless coroutine),其状态由编译器自动生成并管理。一个函数只要包含co_awaitco_yieldco_return中的任意一个,就成为协程。

协程返回类型必须满足“可等待”(awaitable)和“可返回”(returnable)的要求,通常需要定义一个带有特定接口的返回类型,例如:

  • promise_type:定义协程内部行为
  • get_return_object():创建返回值对象
  • initial_suspend():决定协程启动时是否立即暂停
  • final_suspend():决定协程结束时是否暂停
  • unhandled_exception():异常处理逻辑

示例:最简单的协程结构

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

#include 
#include 

struct SimpleTask { struct promise_type { SimpleTask get_return_object() { return {}; } std::suspend_never initial_suspend() { return {}; } std::suspend_never final_suspend() noexcept { return {}; } void return_void() {} void unhandled_exception() {} }; };

SimpleTask hello_coroutine() { std::cout << "Hello from coroutine!\n"; co_return; }

使用 co_await 实现异步等待

co_await用于挂起协程直到某个操作完成,常用于异步I/O、定时任务或任务调度中。要使一个类型可被co_await,它必须实现await_readyawait_suspendawait_resume方法。

示例:自定义可等待对象

struct Awaitable {
    bool await_ready() { return false; } // 返回true则不挂起
    void await_suspend(std::coroutine_handle<> handle) {
        // 可以在这里安排恢复时机,比如加入事件循环
        handle.resume(); // 立即恢复(模拟同步)
    }
    int await_resume() { return 42; }
};

SimpleTask async_example() { int result = co_await Awaitable{}; std::cout << "Received: " << result << "\n"; }

上面的例子中,await_ready返回false表示需要挂起,await_suspend接收协程句柄并控制何时恢复,await_resume返回最终结果。

协程在异步编程中的应用

协程非常适合替代回调地狱式的异步代码。你可以把复杂的异步流程写成类似同步的形式,提升可读性和维护性。

播记
播记

播客shownotes生成器 | 为播客创作者而生

下载

常见应用场景包括:

  • 网络请求等待响应
  • 文件读写完成通知
  • 定时器触发
  • 多阶段任务流水线

示例:模拟异步延迟任务

#include 
#include 
#include 

struct DelayAwaiter { int ms; bool await_ready() { return false; } void await_suspend(std::coroutine_handle<> h) { std::jthread t([h, ms = ms](std::stop_token st) { if (st.stop_requested()) return; std::this_thread::sleep_for(std::chrono::milliseconds(ms)); h.resume(); }); t.detach(); // 注意:生产环境应更安全地管理生命周期 } void await_resume() {} };

SimpleTask delayed_task() { std::cout << "Start\n"; co_await DelayAwaiter{1000}; std::cout << "After 1 second\n"; }

这个例子展示了如何通过co_await实现非阻塞延时,协程在等待期间释放线程资源。

协程返回值与生成器模式

除了异步任务,协程也适合实现生成器(generator)。使用co_yield可以在每次调用时产生一个值,并保持局部状态。

示例:整数序列生成器

struct Generator {
    struct promise_type {
        int current_value;
        Generator get_return_object() { return Generator{this}; }
        std::suspend_always initial_suspend() { return {}; }
        std::suspend_always final_suspend() noexcept { return {}; }
        std::suspend_always yield_value(int value) {
            current_value = value;
            return {};
        }
        void return_void() {}
        void unhandled_exception() { std::terminate(); }
    };
using handle_type = std::coroutine_handle;
explicit Generator(promise_type* p) : coro(handle_type::from_promise(*p)) {}
~Generator() { if (coro) coro.destroy(); }

int value() const { return coro.promise().current_value; }
bool move_next() { 
    if (!coro.done())
        coro.resume();
    return !coro.done();
}

private: handle_type coro; };

Generator range(int from, int to) { for (int i = from; i

// 使用方式 void test_generator() { auto gen = range(1, 5); while (gen.move_next()) { std::cout

这种生成器避免了构建完整容器的开销,适用于大数据流或无限序列。

基本上就这些。C++20协程虽然强大,但需要手动实现promise和awaiter逻辑,有一定学习成本。不过一旦掌握,就能写出清晰高效的异步代码。注意资源管理和生命周期控制,避免悬空句柄或内存泄漏。配合现代事件循环框架(如liburing、folly等),协程将成为高性能服务端编程的重要工具

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

83

2023.09.25

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

533

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

51

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

193

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

95

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1010

2023.10.19

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

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

25

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Node.js 教程
Node.js 教程

共57课时 | 8.3万人学习

CSS3 教程
CSS3 教程

共18课时 | 4.4万人学习

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

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