0

0

c++如何实现生产者消费者模型_c++ 互斥锁与条件变量协同工作【案例】

尼克

尼克

发布时间:2026-01-02 15:51:14

|

441人浏览过

|

来源于php中文网

原创

必须用 std::condition_variable 配合 std::mutex 实现生产者消费者模型,因互斥锁无法解决空/满队列时的线程等待问题;wait() 必须在 unique_lock 持有锁时调用,且需用 while 循环或带谓词的重载检查条件,避免虚假唤醒。

c++如何实现生产者消费者模型_c++ 互斥锁与条件变量协同工作【案例】

std::mutexstd::condition_variable 实现生产者消费者模型,是 C++ 多线程中最典型、也最容易出错的协同场景。核心在于:锁只保护临界区,而条件变量负责“等某个状态成立”,二者必须配合使用,且 wait() 调用前必须已持有锁

为什么不能只用互斥锁?

互斥锁能阻止多个线程同时访问共享队列,但无法解决“队列空时消费者该干什么”或“队列满时生产者该干什么”的问题。轮询(busy-wait)浪费 CPU;单纯加锁后 return 又会导致线程退出,无法持续工作。这时候必须靠 std::condition_variable 挂起线程,等条件满足再唤醒。

std::condition_variable::wait() 的正确调用姿势

这是最常踩坑的地方:wait() 必须在已锁定的 std::unique_lock<:mutex> 上调用,且内部会自动释放锁并挂起线程;被唤醒后,它会重新获取锁,再返回 —— 但此时条件未必仍成立(可能被其他线程抢走资源),所以必须用 while 循环判断,不能用 if。

  • 错误写法:if (queue.empty()) cv.wait(lock); → 唤醒后不检查,可能读空队列
  • 正确写法:while (queue.empty()) cv.wait(lock);
  • wait() 的第二个参数可以是 lambda,等价于 while + lambda 调用,更简洁:cv.wait(lock, [&]{ return !queue.empty(); });

生产者与消费者的完整协作逻辑

假设共享一个 std::queue,容量上限为 10:

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

Amazon Nova
Amazon Nova

亚马逊云科技(AWS)推出的一系列生成式AI基础模型

下载
#include 
#include 
#include 
#include 
#include 

std::queue q; std::mutex mtx; std::condition_variable cv_not_full, cv_not_empty; const int MAX_SIZE = 10;

void producer(int id) { for (int i = 0; i < 5; ++i) { std::unique_lock lock(mtx); cv_not_full.wait(lock, [&]{ return q.size() < MAX_SIZE; }); q.push(i + id 10); std::cout << "P" << id << " pushed " << (i + id 10) << ", size=" << q.size() << "\n"; lock.unlock(); // 手动解锁,避免阻塞消费者 cv_not_empty.notify_one(); // 通知至少一个消费者 } }

void consumer(int id) { for (int i = 0; i < 5; ++i) { std::unique_lock lock(mtx); cv_not_empty.wait(lock, [&]{ return !q.empty(); }); int val = q.front(); q.pop(); std::cout << "C" << id << " popped " << val << ", size=" << q.size() << "\n"; lock.unlock(); cv_not_full.notify_one(); // 通知至少一个生产者 } }

注意两点:一是 notify_one() 在锁外调用更安全(避免唤醒后立即竞争锁);二是两个条件变量分别对应“非满”和“非空”,不能共用一个 —— 否则 notify_all() 会同时唤醒所有生产者和消费者,造成无谓竞争。

容易忽略的边界与性能点

实际部署时这几个细节常被跳过:

  • 共享队列未加 std::atomic 或锁保护,直接裸读写 → 数据竞争,UB
  • 忘记在 wait() 的 lambda 中捕获引用(如 [&]),导致读到的是旧值或编译失败
  • std::lock_guard 替代 std::unique_lockwait() 编译不过,因为后者需要支持中途释放/重获锁
  • 生产者 push 后只调用 notify_one(),但如果有多个消费者等待,可能唤醒的是刚处理完的线程,而其他线程还在等 —— 此时用 notify_all() 更稳妥(代价是唤醒开销略高)

真正难的不是写出来,而是想清楚「谁在等什么条件」「谁来通知」「通知之后是否真能满足」—— 这三问没理清,wait() 就会永远挂住,或者频繁虚假唤醒。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

713

2023.08.22

while的用法
while的用法

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

81

2023.09.25

lambda表达式
lambda表达式

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

202

2023.09.15

python lambda函数
python lambda函数

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

189

2025.11.08

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

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

473

2023.08.10

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

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

131

2025.12.24

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

129

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

75

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

81

2025.12.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
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号