0

0

c++中如何实现生产者消费者模型_c++多线程条件变量应用实例【汇总】

穿越時空

穿越時空

发布时间:2026-01-18 13:15:08

|

209人浏览过

|

来源于php中文网

原创

c++kquote>必须用wait()谓词重载版本防虚假唤醒;notify_all()适用于非独占资源如缓冲区空满;锁须包裹条件检查与操作全过程;条件变量生命周期须长于等待线程。

c++中如何实现生产者消费者模型_c++多线程条件变量应用实例【汇总】

std::mutex + std::condition_variable 实现生产者消费者模型是 C++ 多线程中最典型也最容易出错的实践之一。核心不在于“能不能跑”,而在于“唤醒是否可靠”“边界是否守得住”“虚假唤醒是否被处理”。下面直奔关键点。

为什么必须用 wait() 的谓词重载版本?

直接调用 cv.wait(lock) 是危险的——它不检查条件是否真正满足,只负责挂起和被唤醒。一旦发生虚假唤醒(spurious wakeup),线程会跳过条件判断直接往下执行,导致读空队列或写满队列。

正确做法是始终使用带谓词的重载:cv.wait(lock, [&]{ return !queue.empty(); });

  • 该重载内部自动循环检查谓词,天然防虚假唤醒
  • 谓词必须捕获所有依赖变量(如 queue),且不能抛异常
  • 不要写成 while(queue.empty()) cv.wait(lock); —— 手动 while 容易漏锁或逻辑错位

notify_one() 还是 notify_all()

多数场景下用 notify_one() 更高效,但前提是:你明确知道每次只该唤醒一个等待者。

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

生产者消费者中,常见错误是:

Replit Agent
Replit Agent

Replit最新推出的AI编程工具,可以帮助用户从零开始自动构建应用程序。

下载
  • 生产者调用 notify_one(),但多个消费者在等——只有一个被唤醒,其余继续挂起,没问题
  • 消费者调用 notify_one(),但多个生产者在等——同理,也 OK
  • 但若队列容量有限(如环形缓冲区),且生产者需等待“有空位”,此时多个生产者可能同时被阻塞;如果只 notify_one,其他生产者永远收不到信号——这时必须用 notify_all()

更稳妥的做法:对“非独占资源”(如缓冲区空/满)统一用 notify_all();对“一对一唤醒”(如任务就绪)可用 notify_one()

如何避免死锁和竞争?

最常踩的坑是锁粒度与条件检查脱节。例如:

std::queue queue;
std::mutex mtx;
std::condition_variable cv;

// ❌ 错误:先检查再加锁,中间存在竞态
if (queue.empty()) {
    std::unique_lock lock(mtx);
    cv.wait(lock, [&]{ return !queue.empty(); }); // 可能永远等下去
}

// ✅ 正确:锁必须包裹整个条件检查 + wait + 操作
std::unique_lock lock(mtx);
cv.wait(lock, [&]{ return !queue.empty(); });
int val = queue.front();
queue.pop();
  • 所有对共享数据(queue)的访问,包括 empty()front()pop(),都必须在 std::unique_lock 保护下
  • wait() 内部会自动释放锁,并在唤醒后重新获取——这个机制不能绕过
  • 不要用 std::lock_guard 替代 std::unique_lock,因为 wait() 要求可转移、可释放的锁

要不要用 std::deque无锁结构替代 std::queue

std::queue 默认基于 std::deque,本身不是线程安全的,所以仍需外部同步。换成 std::deque 并不会减少锁需求——只是让你多一层手动管理 push_back()/pop_front() 的麻烦。

真要优化性能,考虑以下路径:

  • 单生产者单消费者(SPSC):可用 boost::lockfree::queue 或自实现环形缓冲区,避开锁
  • 多生产者多消费者(MPMC):标准库无原生支持,moodycamel::ConcurrentQueue 是较成熟选择
  • 但除非压测确认锁是瓶颈,否则别过早替换——std::mutex + std::condition_variable 在现代 glibc/libstdc++ 上已足够高效

真正容易被忽略的是:条件变量的生命周期必须长于所有等待它的线程。如果主线程析构了 cv,而子线程还在 wait(),行为未定义——务必确保同步对象的销毁顺序。

相关专题

更多
while的用法
while的用法

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

89

2023.09.25

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

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

481

2023.08.10

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

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

143

2025.12.24

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

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

143

2025.12.24

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

84

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

24

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

35

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

16

2026.01.15

热门下载

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

精品课程

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