0

0

如何在C++中利用协程实现简单的生产者消费者?(并发模型)

冰火之心

冰火之心

发布时间:2026-03-09 13:31:16

|

908人浏览过

|

来源于php中文网

原创

协程不能直接替代线程实现生产者消费者模型,因其无内置调度与同步机制,需配合condition_variable等手动实现等待/唤醒逻辑。

如何在c++中利用协程实现简单的生产者消费者?(并发模型)

协程不能直接替代线程做生产者消费者

标准 C++20 协程本身不带调度、不带同步原语,co_await 不会自动切到另一个线程,也不会帮你锁队列。你写一个 producer() 和一个 consumer() 协程,它们默认在同一个线程里串行执行——除非你主动把它们交给某个调度器(比如 std::jthread 或第三方库),否则根本谈不上“并发模型”。

常见错误现象:consumer() 一直 co_await 等数据,但 producer() 根本没机会跑;或者两个协程都卡在 co_await 上,程序停住。

  • 协程只是可挂起的函数,不是线程,也不自带唤醒机制
  • 要用协程实现生产者消费者,必须搭配某种等待/通知机制(比如 std::condition_variable)或自定义 awaiter
  • 如果目标是简化并发逻辑,优先考虑 std::thread + std::queue + std::mutex,更直观、更可控

用 std::condition_variable 配合协程做同步

最务实的做法:让协程挂起时交出控制权,等条件满足再被唤醒。这需要自己写一个能响应 std::condition_variableawaiter,比如 sync_waiter

使用场景:你已有线程池或主循环,想把阻塞等待换成协程挂起,避免线程空转。

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

  • std::condition_variable::wait() 必须配合 std::unique_lock<:mutex></:mutex>,协程 awaiter 里也要封装这层逻辑
  • 别直接 co_await cv——std::condition_variable 不是 awaitable,要包装成支持 await_ready()/await_suspend() 的类型
  • 唤醒后要重新检查条件(spurious wakeup),所以 await_resume() 通常返回 bool 或 void,逻辑仍需写在循环里

简短示例(核心结构):

有一导航
有一导航

有一导航延续了美国Groupon网站一贯的简约风格和购物流程,致力于打造中国本土化的精品消费限时团购网站,您会发现网站的页面非常简单,简单到每天只有一款产品。 产品通常不是实物,而是生活消费领域的各类服务型产品,比如服装、饰品、数码、化妆品、培训、健身等各类商品,用户只需在线购买,三分钟就可轻松买到超低折扣的团购产品!

下载
struct sync_waiter {
    std::condition_variable& cv;
    std::mutex& mtx;
    std::atomic<bool>& ready;

    bool await_ready() const noexcept { return ready.load(); }
    void await_suspend(std::coroutine_handle<> h) {
        std::thread([&, h] { 
            std::unique_lock l(mtx); 
            cv.wait(l, [&]{ return ready.load(); }); 
            h.resume(); 
        }).detach();
    }
    void await_resume() const noexcept {}
};

为什么不用 boost::asio 或 libunifex?

因为它们确实能跑通,但代价是引入整套异步运行时。如果你只是想做个玩具级的生产者消费者 demo,用 boost::asio::io_context 启动两个协程,会发现:启动开销大、调试困难、堆分配多、编译时间长。

参数差异明显:boost::asio::use_awaitable 要求所有 I/O 对象都绑定到同一个 io_context,而你的队列是内存对象,没法直接“注册”进去;得额外包一层 postdispatch 才能触发唤醒。

  • libunifex::just_done()libunifex::schedule() 这类操作符看起来简洁,但底层依赖 scheduler 模型,和传统线程模型混用容易出竞态
  • Windows 下 boost::asio 默认用 IOCP,Linux 下用 epoll,行为不一致,调试时容易误判是逻辑问题还是调度问题
  • 若项目不允许外部依赖,硬上这些库反而增加维护成本

真正轻量可行的协程方案:只协程化“等待”,不动线程模型

把生产者和消费者逻辑保留在普通线程里,只把“等数据”和“等空间”这两个阻塞点协程化。这样既利用了协程的可读性,又规避了调度复杂度。

性能影响很小:一次 co_await 就是一次指针保存+跳转,比 pthread_cond_wait() 的系统调用便宜得多;兼容性也好,C++20 编译器基本都支持。

  • 共享队列仍用 std::queue + std::mutex,不变
  • 消费者线程里写 while (true) { auto item = co_await pop_if_available(); ... },其中 pop_if_available 是一个返回 std::optional<t></t> 的协程
  • 关键点:pop_if_available 内部用 try_lock,失败就 co_await 一个短暂休眠(如 std::this_thread::sleep_for(1ms)),而不是死等

这个方案不会让你代码变“高大上”,但它跑得稳、看得懂、改得快。

最容易被忽略的是:协程栈生命周期。如果你在协程里捕获了局部变量的引用,而该协程被挂起后局部变量已析构,后续 resume 就是未定义行为——这点比线程更隐蔽,也更难 debug。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
while的用法
while的用法

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

105

2023.09.25

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

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

186

2023.11.23

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

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

130

2025.11.27

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

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

438

2023.07.18

堆和栈区别
堆和栈区别

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

601

2023.08.10

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

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

438

2023.07.18

堆和栈区别
堆和栈区别

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

601

2023.08.10

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

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

764

2023.08.10

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

热门下载

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

相关下载

更多

精品课程

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

共94课时 | 11万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.2万人学习

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

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