0

0

C++ 条件变量(Condition Variable)是什么?(如何解决生产者消费者问题)

冰火之心

冰火之心

发布时间:2026-02-17 17:57:51

|

682人浏览过

|

来源于php中文网

原创

condition_variable必须与mutex配合使用,因其自身不保证线程安全,仅负责等待与唤醒;所有共享状态的读写需由同一mutex串行化,且wait前必须加锁、条件检查与wait共用同一mutex。

c++ 条件变量(condition variable)是什么?(如何解决生产者消费者问题)

condition_variable 为什么必须和 mutex 一起用

因为 condition_variable 本身不保证线程安全,它只负责“等待”和“唤醒”,不保护共享数据。所有对共享状态(比如缓冲区是否为空、是否满)的读写,都得靠 mutex 串行化。漏掉锁,wait() 前的状态检查就可能被其他线程改写,导致虚假唤醒或死锁。

  • wait() 内部会自动释放传入的 mutex,等被唤醒后再重新加锁——这是它和普通 while 循环+sleep 的本质区别
  • 必须在 wait() 前加锁,且锁的对象要和检查条件用的是同一个 mutex
  • 别用 std::atomic 替代 mutex:原子变量能保单个读写,但“检查+等待”是两步操作,中间有竞态窗口

notify_one() 和 notify_all() 到底该选哪个

看等待线程是否“等同”。生产者消费者模型里,多个消费者都在等“有数据”,一个数据来了只够一个消费者取,用 notify_one() 更高效;但如果所有等待者都需要响应同一事件(比如关闭信号),就得用 notify_all()

  • notify_one() 不保证唤醒谁,但避免了“惊群”——100 个消费者同时被唤醒却只有一个能干活,其余又立刻回去等
  • notify_all() 要求所有等待线程的条件判断必须用 while 而不是 if,否则可能因虚假唤醒读到无效状态
  • 即使只调用了 notify_one(),也要用 while 循环检查条件:C++ 标准允许系统在无通知时唤醒线程(spurious wakeup)

生产者消费者代码里最容易错的三处

不是逻辑写错,而是同步原语用法细节没抠准。下面这三行写错任意一个,程序就可能卡死或崩溃:

Yourware
Yourware

专注于AI编程作品部署与分享的云托管平台

下载
  • 消费者中写成 if (buffer.empty()) cv.wait(lock) —— 必须是 while,不是 if
  • 生产者 push 后只调用 cv.notify_one(),但消费者在等“非空”,而生产者可能 push 多次后才 notify,中间一次 notify 丢了就永远等下去——必须每次修改状态后 notify
  • cv.wait(lock, []{ return !buffer.empty(); }); 的 lambda 捕获写成 [&buffer],但 buffer 是局部变量或生命周期短的对象,lambda 执行时已析构

std::condition_variable 和 std::condition_variable_any 的区别在哪

std::condition_variable 只认 std::mutex 及其兼容类型(如 std::recursive_mutex);std::condition_variable_any 能配合任意满足 Lockable 概念的锁,比如 std::shared_mutex 或自定义锁。但代价是性能差不少——内部多一层类型擦除。

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

  • 日常生产者消费者,用 std::condition_variable + std::mutex 就够了,别为了“通用”选 any 版本
  • std::condition_variable_anywait() 在异常抛出时可能不会自动重锁,需要更小心的异常安全处理
  • Windows 上 std::condition_variable 底层用的是 SRWLock + WaitOnAddress,Linux 上是 futex,都不支持任意锁类型——所以 any 版本其实是模拟实现,开销实打实
实际写的时候,最常被忽略的是:条件变量不保存状态,它只是个信号通道。buffer.size() 变了,你得自己记、自己查、自己 notify。没人替你记住“刚才有没有 notify 过”。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

816

2023.08.22

while的用法
while的用法

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

103

2023.09.25

lambda表达式
lambda表达式

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

212

2023.09.15

python lambda函数
python lambda函数

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

192

2025.11.08

Python lambda详解
Python lambda详解

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

58

2026.01.05

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

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

675

2023.08.10

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

1137

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1150

2023.07.27

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

462

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.6万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.2万人学习

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

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