新闻中心
C++如何使用条件变量_C++多线程同步std::condition_variable的等待与通知
std::condition_variable用于线程同步,配合unique_lock实现等待通知机制。生产者修改共享状态并调用notify唤醒消费者,消费者通过wait阻塞直至条件满足。使用谓词可避免虚假唤醒,确保数据一致性与线程安全。

在C++多线程编程中,std::condition_variable 是实现线程间同步的重要工具之一。它通常与 std::unique_lock<:mutex> 配合使用,用于阻塞一个或多个线程,直到另一个线程修改了某个共享状态并通知它们继续执行。
条件变量的基本原理
条件变量允许线程在某个条件不满足时进入等待状态,避免频繁轮询消耗CPU资源。当其他线程改变了这个条件后,通过发送“通知”来唤醒等待中的线程。
核心机制包括:
- wait():使当前线程阻塞,直到被唤醒
- notify_one():唤醒一个正在等待的线程
- notify_all():唤醒所有等待的线程
基本用法:等待与通知
以下是一个典型的生产者-消费者模型示例,展示如何使用 std::condition_variable 实现线程同步:
立即学习“C++免费学习笔记(深入)”;
晓象AI资讯阅读神器
晓象-AI时代的资讯阅读神器
72
查看详情
#include <iostream>
#include <thread>
#include <queue>
#include <mutex>
#include <condition_variable>
std::queue<int> data_queue;
std::mutex mtx;
std::condition_variable cv;
bool finished = false;
void producer() {
for (int i = 0; i < 5; ++i) {
std::this_thread::sleep_for(std::chrono::milliseconds(100));
std::unique_lock<std::mutex> lock(mtx);
data_queue.push(i);
std::cout << "生产: " << i << "\n";
lock.unlock();
cv.notify_one(); // 通知消费者
}
{
std::lock_guard<std::mutex> lock(mtx);
finished = true;
}
cv.notify_all(); // 通知所有消费者结束
}
void consumer() {
while (true) {
std::unique_lock<std::mutex> lock(mtx);
// 条件判断:队列为空且未结束,则等待
cv.wait(lo
ck, []{ return !data_queue.empty() || finished; });
if (!data_queue.empty()) {
int value = data_queue.front();
data_queue.pop();
std::cout << "消费: " << value << "\n";
}
if (data_queue.empty() && finished) {
break; // 结束循环
}
lock.unlock();
}
}
在上面的例子中:
- 生产者每隔一段时间向队列添加数据,并调用 notify_one()
- 消费者调用 wait(),传入锁和一个lambda表达式作为谓词(predicate)
- 只有当谓词返回 true 时,wait 才会解除阻塞
- 使用谓词可以防止虚假唤醒导致的问题
关键注意事项
正确使用 std::condition_variable 需要注意以下几点:
- 必须配合 std::unique_lock<:mutex> 使用,不能用 lock_guard
- wait 调用会自动释放锁,在唤醒后重新获取锁,保证安全性
- 始终使用带谓词的 wait 形式(即 wait(lock, predicate)),避免虚假唤醒问题
- 通知方修改共享数据时也应持有同一互斥锁,确保数据一致性
- 如果可能唤醒多个消费者,考虑使用 notify_all() 替代 notify_one()
基本上就这些。合理使用条件变量可以让多线程程序更高效、响应更及时,同时避免忙等待带来的性能浪费。掌握 wait 和 notify 的配对逻辑是关键。
以上就是C++如何使用条件变量_C++多线程同步std::condition_variable的等待与通知的详细内容,更多请关注其它相关文章!
# 如何实现
# 常见网站推广方式珍藏
# 如何找文章的关键词排名
# 咖啡的营销推广字体设计
# 安徽营销推广哪家好
# 巩义搜狗网站推广
# 关键词查询网站排名工具
# 营销年度推广计划
# 网站建设建站多久出证
# 片子网站建设美丽文案
# 山西seo如何做
# 才会
# 是一个
# 工具
# 最常用
# 有什么区别
# 尼克
# 软件工程
# 多个
# 如何使用
# 多线程
# red
# stream
# ios
# c++
# ai
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
win11 Snap Layouts怎么用 Win11窗口布局与分屏多任务高效指南【必学】
2025AO3夸克浏览器通道_AO3手机HTTPS安全入口分享
CSS自定义字体样式被系统字体替换怎么办_font-face方式指定font-display控制渲染策略
Pandas DataFrame 高效批量赋值:告别循环与笛卡尔积误区
PHP高效扁平化嵌套数组:使用array_merge与数组解包操作符
将HTML动态表格多行数据保存到Google Sheet的教程
C++ explicit关键字防止隐式转换_C++构造函数安全规范
处理动态列数据:J*a ArrayList的正确初始化与字符累加教程
如何将HTML表格多行数据保存到Google Sheets
TikTok国际版网页端快速入口 TikTok全球版短视频浏览教程
优化LangChain文档加载与ChromaDB集成:解决多文档处理与分块问题
京东单号查询入口_京东快递订单追踪入口
知音漫客官网漫画下载_知音漫客网页版阅读记录
CSS Flexbox如何实现多行排列_flex-wrap wrap自动换行显示
Sublime怎么配置Nim语言环境_Sublime Nim代码高亮与补全
Golang如何使用buffered channel提高性能_Golang buffered channel优化技巧
如何在Python中使用Optional类型处理可变对象并避免Pylint警告
拼多多赚钱渠道_拼多多收益来源
React列表渲染与独立状态管理:避免全局状态影响局部更新
小红书网页版入口链接分享 小红书官网直接进
蛙漫移动版在线看 蛙漫手机浏览器直达入口
在J*a中如何使用Stream.map转换元素_Stream映射操作解析
如何仅使用CSS更改登录界面背景图像图标的颜色
如何在CSS中使用浮动制作导航栏_float实现水平菜单
Python中高效访问嵌套字典与列表中的键值对
如何在J*a中实现统一对象行为接口_项目大型化时的接口规范化
Go语言中对Map值调用带指针接收者方法:原理与最佳实践
利用Bokeh CustomJS动态控制DataTable列可见性
windows10怎么查看硬盘序列号_windows10硬盘id查询命令
Win10磁盘清理工具在哪 Win10打开并使用磁盘清理【教程】
PS5 Pro有点优势但不多! 《燕云十六声》PS5平台与PC性能画面对比
Fabric模组开发:自定义物品与物品组的现代管理方法
如何在Promise链中有效终止错误处理后的执行
响应式CSS Grid布局:优化网格项在小屏幕下的堆叠与宽度适配
Spring Boot内嵌服务器与J*a EE全栈特性:选择与部署策略
ArrayList与LinkedList核心操作的Big-O复杂度分析
CSS布局中意外空白:解决padding-top导致的顶部间距问题
Win10自动更新怎么关闭 Win10永久关闭系统更新的两种方法【终极版】
反效果?《战地6》免费试玩开启后玩家数不升反降
Yandex搜索引擎官方地址 俄罗斯网络世界的主要入口
Go语言JSON解析深度指南:动态访问与结构体映射实践
Win11怎么修改默认浏览器_Windows 11设置Chrome为默认
优化 Jest 模拟:强制未实现函数抛出错误以提升测试效率
Win11怎么查看显卡显存 Win11显示适配器属性及专用视频内存查询
邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策
马斯克:Optimus 人形机器人复数形式为 Optimi
UC浏览器官网入口2025最新 UC浏览器网页版正式地址
Discord Slash 命令响应超时问题的异步解决方案
单射、满射与双射的关系 一文理清所有逻辑
如何使用spryker/configurable-bundles-products-resource-relationship模块解决复杂产品捆绑关系难题


2025-12-16
浏览次数:次
返回列表
ck, []{ return !data_queue.empty() || finished; });
if (!data_queue.empty()) {
int value = data_queue.front();
data_queue.pop();
std::cout << "消费: " << value << "\n";
}
if (data_queue.empty() && finished) {
break; // 结束循环
}
lock.unlock();
}
}