0

0

C++中的co_return是什么?(如何在协程中返回最终结果)

冰火之心

冰火之心

发布时间:2026-03-01 12:20:02

|

273人浏览过

|

来源于php中文网

原创

co_return是协程专用返回语句,需配合promise_type使用,触发return_void()/return_value()并进入final_suspend状态,而非直接返回调用方。

c++中的co_return是什么?(如何在协程中返回最终结果)

co_return 是协程的返回语句,不是普通函数 return 的替代品

它只在标记为 coroutine 的函数(即含 co_awaitco_yieldco_return 的函数)中合法,且必须配合自定义的协程桩(promise_type)才能工作。编译器看到 co_return 会生成对 promise.return_void()promise.return_value(...) 的调用,而不是直接结束函数。

常见错误现象:error: 'co_return' cannot be used in a function that is not a coroutine——说明函数没被识别为协程,通常因为没用到其他协程关键字,或编译器未启用 C++20 协程支持(如没加 -std=c++20-fcoroutines)。

  • 使用场景:异步 I/O 完成后返回结果(如 Task<int></int>)、生成器末尾收尾(Generator<void></void>)、资源清理后退出协程
  • 若协程返回类型是 void(如 Task<void></void>),必须调用 promise.return_void();若返回非 void(如 int),则调用 promise.return_value(val)
  • 不写 co_return 也不报错,但协程会在函数末尾隐式调用 return_void();显式写更清晰,尤其需要提前退出时

co_return 后协程不立即销毁,而是进入 final_suspend 状态

这是最容易被忽略的一环:co_return 执行完,协程帧(coroutine frame)不会自动释放,而是先挂起在 promise.final_suspend() 返回的 awaiter 上。是否继续执行、何时销毁,全由这个 awaiter 决定。

典型陷阱:忘了重载 final_suspend,或返回了 std::suspend_always{} 却没手动 destroy(),导致协程泄漏、内存不释放。

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

HaloTool
HaloTool

AI工具在线集合网站

下载
  • 如果希望协程结束后立刻销毁(最常见需求),final_suspend() 应返回 std::suspend_never{}
  • 如果要移交控制权(比如把协程句柄交给 event loop 处理),可返回 std::suspend_always{},并在外部调用 handle.destroy()
  • 返回 std::suspend_always{} 时,handle.done() 会返回 true,但 handle.destroy() 必须由用户显式调用,否则协程帧永远驻留

co_return 表达式的类型必须能被 promise_type 接受

co_return expr; 中的 expr 类型,必须和 promise_type::return_value(T&&) 的参数类型匹配(或能隐式转换)。否则编译失败,错误信息往往指向 promise 类型里缺失该函数。

例如:声明 Task<:string></:string> 却在协程里写 co_return 42;,就会触发类似 no matching member function for call to 'return_value' 的错误。

  • 基本类型(intstd::string)通常走 return_value(T&&)
  • void 协程只能用 co_return;(无表达式),触发 return_void()
  • 如果想支持多种返回方式(如允许 co_returnco_return std::move(x)),需同时提供 return_value(const T&)return_value(T&&)
  • 注意移动语义:co_return local_obj; 默认触发移动(如果 T 可移动),但若 return_value 参数是 const T&,就只能拷贝

别把 co_return 当 return 用——它不传递值给调用方

co_return 的值不是“返回给调用者”,而是传给协程的 promise,再由 promise 决定如何暴露(比如存入 Task 对象内部、触发回调、填充 future)。调用协程函数得到的只是句柄(std::coroutine_handle)或包装对象(如 Task<int></int>),真正取结果得靠后续机制(如 co_awaitget())。

新手常误以为 auto x = my_coro(); co_return 123; 会让 x 直接等于 123,其实 x 是个待调度的协程对象,里面还没运行,更没结果。

  • 同步获取结果?不行——协程默认惰性求值,必须驱动它(如 co_await xx.get(),后者通常阻塞等待)
  • 想让 co_return val 最终变成调用方拿到的 val,必须确保 promise 正确存储 val,且 awaiter 的 await_resume() 返回它
  • 性能影响:每次 co_return 都涉及一次 promise.return_value() 调用 + 可能的移动/拷贝;避免在热路径频繁返回大对象
协程的返回逻辑绕在 promise 里,不是语法糖,也不是控制流终点。漏掉 final_suspend 或写错 return_value 签名,编译期可能不报错,但运行时行为完全不可控。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

890

2023.08.02

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

411

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

357

2023.10.25

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

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

557

2023.09.20

string转int
string转int

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

890

2023.08.02

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

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

595

2024.08.29

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

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

294

2025.08.29

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

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

210

2025.08.29

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

6

2026.02.28

热门下载

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

精品课程

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

共94课时 | 10.4万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 19.8万人学习

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

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