c++++20协程是一种控制流机制,非并发原语,通过co_await、co_yield和co_return关键字实现挂起与恢复。1. 协程基于无栈设计,由编译器生成状态机管理上下文;2. 挂起时通过awaiter的await_ready()、await_suspend()、await_resume()控制执行转移;3. 无栈特性节省内存并减少上下文切换开销;4. 自定义awaiter和promisetype可扩展性强,但需注意生命周期和线程安全。

C++20的协程特性,简而言之,为异步编程提供了一种更自然、更易读的语法糖。它允许函数在执行过程中“挂起”,并在需要时从中断点“恢复”,而这一切都建立在“无栈”的底层实现之上,即协程不会像传统线程那样拥有独立的运行时栈,而是通过编译器生成的状态机来管理其上下文和局部变量。挂起函数是这一机制的核心,它定义了协程何时以及如何暂停执行的逻辑。

理解C++20协程,我们首先要抓住其核心理念:它不是一个新的并发原语,而是一种控制流机制,旨在简化异步代码的编写。传统的异步回调地狱或Future/Promise链式调用,在协程这里被扁平化为看似同步的顺序代码。co_await、co_yield和co_return这三个关键字,是编译器识别协程的标志,也是我们与协程机制交互的接口。当编译器看到这些关键字时,它会将我们的协程函数转换为一个状态机,负责保存和恢复协程的内部状态。
协程的“挂起”并非代码执行的完全停止,而是一种控制权的转移。具体来说,当一个协程遇到co_await表达式时,它会尝试等待一个“可等待对象”(Awaitable)。这个可等待对象内部有一个“等待器”(Awaiter),它定义了协程如何进行挂起和恢复的逻辑。
立即学习“C++免费学习笔记(深入)”;

一个典型的等待器需要实现三个关键成员函数:
await_ready():这是一个快速检查,用于判断是否需要真正挂起。如果返回true,表示结果已经就绪,协程将不会挂起,而是直接执行await_resume()并继续。这对于避免不必要的上下文切换非常有用。await_suspend(std::coroutine_handle<Promise> handle):这是真正的挂起操作发生的地方。当await_ready()返回false时,协程会调用此函数。在这里,等待器可以将当前的协程句柄(handle)保存起来,然后将控制权交还给调用者或调度器。await_suspend可以返回void(立即恢复调用者)、bool(true表示挂起,false表示不挂起并立即恢复调用者),或者一个std::coroutine_handle<>(将控制权转移给另一个协程)。这个函数执行完毕后,当前协程就处于挂起状态。await_resume():当协程被外部机制(例如调度器)恢复时,会调用此函数。它负责获取等待操作的结果,并将其返回给co_await表达式。所以,“挂起”的本质,就是通过await_suspend将当前协程的执行上下文(通过coroutine_handle表示)保存起来,并把CPU的控制权交出去。当异步操作完成时,外部机制会通过之前保存的coroutine_handle来调用await_resume,让协程从上次中断的地方继续执行。这就像你正在读一本书,读到一半突然有事,你放了个书签(保存了coroutine_handle),去处理别的事情,等事情办完了,你再翻到书签那里继续阅读。

传统线程为了能够独立运行,需要分配自己的栈空间来存储局部变量、函数调用栈帧以及返回地址等信息。这个栈空间通常不小,而且线程间的上下文切换涉及到内核模式的切换,开销相对较大。这就是为什么大量线程会迅速耗尽系统资源并导致性能下降的原因。
无栈协程则完全不同。它的“无栈”指的是协程自身不维护一个独立的、动态增长的运行时栈。相反,当编译器将一个协程函数转换为状态机时,它会分析协程的生命周期,并为其生成一个“协程帧”(coroutine frame)。这个协程帧是一个结构体,它包含了协程在挂起时需要保存的所有信息:包括局部变量、参数、返回地址以及内部的状态机变量。
这个协程帧通常是在堆上分配的(尽管也可以通过promise_type的自定义分配器进行优化,甚至在某些情况下可以栈分配),而不是像线程栈那样由操作系统预先分配一块大内存。当协程挂起时,其所有相关状态都保存在这个协程帧中。当协程恢复时,执行流直接跳回到这个协程帧对应的位置,继续执行。
这种设计带来了显著的优势:
正是因为这些特性,无栈协程特别适合于I/O密集型任务,例如网络服务器,可以创建成千上万个协程来处理并发连接,而无需为每个连接分配一个独立的线程。
C++20协程的强大之处在于其高度的可定制性。通过实现自定义的Awaiter和PromiseType,我们可以将协程无缝集成到各种异步框架和调度模型中。但这并非没有挑战,需要对协程的生命周期和内存管理有深入的理解。
自定义Awaiter:
当你想要co_await一个非协程库提供的类型时,你需要为它实现await_ready、await_suspend和await_resume。
await_ready():这是性能优化的第一道防线。如果你的异步操作可能立即完成(例如,缓存命中、数据已在内存中),那么await_ready()应该返回true,避免不必要的挂起和恢复开销。await_suspend(std::coroutine_handle<Promise> handle):这是核心。在这里,你需要将handle保存下来,并启动你的异步操作。当异步操作完成时,你需要一个机制来通知协程并调用handle.resume()。这通常涉及到回调函数、事件队列或者将handle注册到某个调度器中。void:表示协程被挂起,控制权返回给调用者,稍后需要手动handle.resume()。bool:true表示挂起,false表示不挂起,协程立即执行await_resume()。这提供了一种条件挂起的机制。std::coroutine_handle<>:这是一种高级用法,允许你在挂起当前协程后,立即恢复另一个协程。这在实现一些复杂的协程调度策略时非常有用。await_resume():这个函数在协程被恢复后执行,通常用于返回异步操作的结果。你需要确保在await_suspend中启动的异步操作的结果,能够安全地传递到await_resume中。异常处理也在这里进行,如果异步操作失败,await_resume可以抛出异常。自定义PromiseType:PromiseType是协程的“管家”,它定义了协程的整体行为和生命周期管理。每个协程函数都会有一个与之关联的PromiseType实例。
get_return_object():协程函数执行后,最终会返回一个对象(例如Task<T>、Future<T>)。这个函数负责创建并返回这个对象,它通常包含coroutine_handle,以便外部可以控制协程。initial_suspend():定义协程在开始执行时是否立即挂起。std::suspend_always:协程在入口点立即挂起,需要外部显式resume才能开始执行。这在实现惰性执行的协程时很有用。std::suspend_never:协程立即开始执行,直到遇到第一个co_await或co_return。final_suspend():定义协程在完成执行(co_return或抛出异常)后是否挂起。std::suspend_always:协程在结束时挂起,允许外部在协程销毁前进行一些清理或观察操作。std::suspend_never:协程在结束时立即销毁。return_void() / return_value(T value):当协程执行co_return;或co_return value;时,会调用这些函数,用于将结果存储到PromiseType中,供get_return_object返回的对象获取。unhandled_exception():当协程内部抛出未捕获的异常时,会调用此函数。你可以在这里处理异常,例如将其存储起来,以便await_resume可以重新抛出,或者进行日志记录。在实践中,自定义Awaiter和PromiseType往往是成对出现的。PromiseType负责管理协程的整体生命周期和结果,而Awaiter则专注于某个特定异步操作的挂起和恢复逻辑。设计时需要特别注意内存所有权、生命周期管理以及线程安全问题,尤其是在跨线程恢复协程的场景下。一个常见的错误是协程句柄在异步操作完成前被销毁,导致野指针或崩溃。深入理解这些机制,是驾驭C++20协程的关键。
以上就是如何理解C++20的协程特性 挂起函数与无栈协程实现的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号