新闻中心

C++ condition_variable条件变量详解_C++线程同步与等待机制

2025-12-15
浏览次数:
返回列表
condition_variable用于线程同步,配合mutex实现等待-通知机制,避免轮询。线程调用wait时释放锁并阻塞,直到notify唤醒;推荐使用带谓词的wait防止虚假唤醒。notify_one唤醒一个线程,notify_all唤醒所有,适用于生产者-消费者等场景。

c++ condition_variable条件变量详解_c++线程同步与等待机制

在C++多线程编程中,condition_variable 是实现线程同步与等待机制的重要工具之一。它通常与 std::mutex 配合使用,用于让一个或多个线程等待某个特定条件成立后再继续执行。这种机制避免了轮询资源状态带来的性能浪费,提高了程序效率和响应性。

condition_variable 的基本概念

condition_variable(条件变量)本身并不存储状态,它只是一个通知机制。线程通过调用其 wait() 方法进入阻塞状态,直到其他线程调用 notify_one()notify_all() 唤醒它。

关键点:

  • 必须配合 std::unique_lock<:mutex> 使用,不能单独使用普通锁。
  • wait 操作会自动释放锁,并在被唤醒后重新获取锁。
  • 常用于生产者-消费者模型、任务队列等场景。

基本用法示例

以下是一个典型的生产者-消费者例子,展示 condition_variable 如何协调线程间通信:

#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::lock_guard<std::mutex> lock(mtx);
            data_queue.push(i);
            std::cout << "Produced: " << i << "\n";
        }
        cv.notify_one(); // 通知一个等待的消费者
    }
    {
        std::lock_guard<std::mutex> lock(mtx);
        finished = true;
    }
    cv.notify_one();
}

void consumer() {
    while (true) {
        std::unique_lock<std::mutex> lock(mtx);
        // 等待条件:队列非空 或 生产结束
        cv.wait(lock, []{ return !data_queue.empty() || finished; });

        if (!data_queue.empty()) {
            int value = data_queue.front();
            data_queue.pop();
            std::cout << "Consumed: " << value << "\n";
        }

        if (data_queue.empty() && finished) {
            std::cout << "No more data. Exiting.\n";
            break;
        }
        // 自动释放锁并继续循环
    }
}

在这个例子中,消费者线程调用 cv.wait() 时会被挂起,直到生产者添加数据并调用 notify_one()。wait 内部使用 lambda 判断条件是否满足,只有当条件为假时才真正阻塞。

AI Code Reviewer AI Code Reviewer

AI自动审核代码

AI Code Reviewer 112 查看详情 AI Code Reviewer

wait 的两种形式

condition_variable 提供了两个版本的 wait 函数:

  • wait(unique_lock& lock):无条件等待,需手动检查条件。
  • wait(unique_lock& lock, Predicate pred):接受一个可调用对象作为谓词,仅当 pred() 返回 false 时才阻塞。

推荐使用带谓词的版本,它可以防止因虚假唤醒(spurious wakeup)导致的问题,确保线程只在真正符合条件时才继续运行。

notify_one 与 notify_all 的区别

  • notify_one():唤醒至少一个正在等待的线程。适用于只有一个线程需要处理任务的情况,如单个消费者。
  • notify_all():唤醒所有等待中的线程。适合广播事件,比如资源已准备好,多个线程都可以继续工作。

注意:即使调用了 notify,等待线程也不会立即运行,因为它必须先重新获取互斥锁。

基本上就这些。掌握 condition_variable 的核心在于理解“等待-通知”模式以及如何安全地配合锁使用。只要注意保护共享数据、正确编写等待条件、及时发送通知,就能写出高效可靠的并发代码。

以上就是C++ condition_variable条件变量详解_C++线程同步与等待机制的详细内容,更多请关注其它相关文章!


# 适用于  # 海参seo  # 高级seo技巧  # 抖音seo 快排  # 青州电商网站建设方案  # 南门专业网站建设  # 娄底租房网站建设文案  # 济宁高级网站建设推广  # SEO怎么优化类目  # 不当推广营销内容是什么  # 福建省网络营销推广app价格表  # 在这个  # 有哪些  # 面试题  # 工具  # 推荐使用  # 挂起  # 多个  # 多线程  # 时才  # 是一个  # red  # 区别  # stream  # ios  # c++  # ai 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: J*a 递归快速排序中静态变量的状态管理与陷阱  汽车之家官方网站官网入口_汽车之家网页版直接进入  Go语言中动态执行代码字符串的策略与实践  c++ 获取系统当前时间 c++时间戳获取方法  护手霜蹭到袖口上了如何清洗? 怎样避免留下一圈油印?  QQ邮箱在线登录平台 QQ邮箱个人邮箱网页版入口  Gmail邮箱申请注册直达_Gmail邮箱免费注册PC版官网入口2025  苹果手机指南针不准怎么校准 传感器校准方法详解【建议收藏】  2306选座时如何选靠窗位置_12306选座靠窗座位查看方法解析  J*aScript数组对象转换:按指定键分组与值收集  如何提高微信支付的安全性_微信支付安全防护与设置建议  漫蛙官网正版漫画入口 漫蛙2官方网页登录地址  必由学官方网站入口 必由学学生教师共用登录通道  b站怎么取消点赞_b站点赞取消操作方法  J*aScript:在map操作中高效处理空数组  Pygame教程:解决用户输入与游戏状态更新不同步问题  J*aScript数据结构转换:将对象数组按类别分组  双系统安装时,如何设置默认启动系统? msconfig命令了解一下!  漫蛙网页登录入口 漫蛙漫画官方授权网址  如何有效阻止外部脚本意外修改内联样式的高度属性  深入理解rpy2中的类型转换:优化Python对象到R矩阵的映射  深入理解Google Cloud Datastore查询:祖先路径与数据一致性  Python模块化编程:有效管理依赖与避免循环引用  J*a最大堆Heapify方法修复:索引计算与边界条件深度解析  QQ邮箱官方登录入口_QQ邮箱网页版快捷使用平台  在J*a中如何使用Stream.map转换元素_Stream映射操作解析  C++如何操作大型数据集_使用C++流式处理(Streaming)技术避免一次性加载大文件  俄罗斯搜索引擎Yandex指南 附2025年免登录官网入口  微博网页版主页入口 微博官方网站免登录访问  Archive of Our Own官网直达 AO3最新可用地址一览  谷歌google账号注册详细步骤 谷歌账号注册官方教程  《燕云十六声》两周内达九百万玩家!位居畅销榜第五  如何在 Excel Online 和 Google 表格中更改日期格式  马斯克:Optimus 人形机器人复数形式为 Optimi  如何在CSS中使用visited与link控制链接颜色_visited link伪类配合  小米14应用无法联网原因分析_小米14网络权限修复  天眼查企业查询官网入口 天眼查官方网页版查询  Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏  移动端XML文件怎么转换成Excel 手机和平板上的解决方案  C++ vector二维数组定义_C++ vector of vector用法  TikTok国际版网页端快速入口 TikTok全球版短视频浏览教程  J*aScript中针对特定容器内图片动画的实现教程  蛙漫限时开放最深处链接_蛙漫全站漫画会员同款秒开地址  快手极速版在线观看 官方网页版登录地址  优化MinIO list_objects_v2 操作的性能瓶颈与最佳实践  天眼查怎么看公司融资情况 天眼查企业融资历史查询步骤【攻略】  sublime如何处理大型CSV文件的列对齐_sublime高级表格编辑插件指南  在J*a中如何隐藏复杂性_使用门面模式组织对象交互  Go语言中对Map值调用带指针接收者方法:原理与最佳实践  Android Studio计算器C键功能异常排查与修复教程 

搜索