0

0

c++中如何实现生产者消费者模型_c++多线程条件变量同步

冰火之心

冰火之心

发布时间:2026-01-14 13:57:38

|

860人浏览过

|

来源于php中文网

原创

std::mutex 无法实现线程等待条件成立,需配合 std::condition_variable 和 std::unique_lock 使用;后者支持自动释放/重获锁,必须用谓词循环检测防止虚假唤醒。

c++中如何实现生产者消费者模型_c++多线程条件变量同步

为什么直接用 std::mutex 不够用

只靠 std::mutex 无法让线程“等某个条件成立”,比如“队列非空”或“队列未满”。你可能会写成忙等待:

while (queue.empty()) { std::this_thread::yield(); }
,这浪费 CPU,且可能因调度延迟导致响应滞后。真正需要的是:线程挂起,直到被明确通知条件满足。

std::condition_variable 必须和 std::unique_lock<:mutex> 配合使用

这是最容易出错的点——std::condition_variable::wait() 要求传入的锁必须是 std::unique_lock(不能是 std::lock_guard),因为 wait() 内部会先释放锁、挂起线程,唤醒后再重新加锁。若用错锁类型,编译直接报错。

典型写法:

std::unique_lock lock(mtx);
cond_var.wait(lock, [&] { return !queue.empty(); }); // 唤醒后自动重加锁
  • lambda 中的判断必须是“实际检查条件”,不能只写 queue.empty() 后取反,否则虚假唤醒会导致逻辑错误
  • 不要在 wait() 外手动 lock.unlock(),它会破坏原子性
  • notify_one()notify_all() 不需要持有锁,但通常建议在持有锁的上下文中调用,确保通知与状态更新的可见性

生产者和消费者都要处理虚假唤醒

即使没人调用 notify_*wait() 也可能返回(POSIX 和 C++ 标准允许)。所以永远要用带谓词的 wait(),或者手动循环检查:

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

std::unique_lock lock(mtx);
while (queue.empty()) {
    cond_var.wait(lock);
}
// 此时 queue 仍可能为空,必须再检查

更安全、更简洁的写法就是用谓词版本:cond_var.wait(lock, [&]{ return !queue.empty(); });,它内部已帮你做了循环。

  • 生产者端同样要检查“是否已满”:cond_var.wait(lock, [&]{ return queue.size()
  • 注意:queue.size() 在多线程下不是原子操作,必须在锁保护下读取

别忘了在析构前停止线程并清理资源

如果生产者/消费者线程还在运行,而容器(如 std::queue)已被析构,就会触发未定义行为。常见做法是引入一个 std::atomic done{false} 控制循环,并在退出前调用 cond_var.notify_all() 唤醒所有等待线程。

例如消费者循环结构:

while (!done.load()) {
    std::unique_lock lock(mtx);
    cond_var.wait(lock, [&]{ return !queue.empty() || done.load(); });
    if (done.load() && queue.empty()) break;
    auto item = std::move(queue.front());
    queue.pop();
    lock.unlock();
    process(item);
}

这里 done.load() 参与了 wait 谓词,确保能及时响应退出信号;同时唤醒后再次检查 done,避免漏掉终止指令。

虚假唤醒、锁类型误用、谓词缺失、析构竞态——这四点覆盖了 90% 的实际 crash 和死锁场景。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

204

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

45

2026.01.05

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

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

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

22

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

18

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

7

2026.01.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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