新闻中心

C++怎么使用条件变量condition_variable_C++线程同步与condition_variable使用

2025-11-15
浏览次数:
返回列表
condition_variable用于线程同步,配合mutex实现等待-通知机制。线程调用wait阻塞自身,直到其他线程调用notify_one或notify_all唤醒;因存在虚假唤醒,需在循环中检查条件谓词。典型步骤包括:定义condition_variable、mutex和共享条件变量;等待线程使用unique_lock加锁并调用wait等待条件满足;通知线程修改条件后调用notify唤醒等待线程。示例为生产者-消费者模型:生产者向队列添加数据并通知,消费者等待队列非空后消费,生产结束后设置finished标志并通知退出。关键点:wait的谓词参数防止虚假唤醒;notify无需持锁但应在修改共享数据后调用以保证可见性;notify_one适用于单线程唤醒,notify_all用于广播但有性能开销。正确使用可避免忙等待,提升效率。

c++怎么使用条件变量condition_variable_c++线程同步与condition_variable使用

在C++多线程编程中,condition_variable 是实现线程间同步的重要工具之一。它通常与 std::mutex 配合使用,用于在线程之间传递状态变化的通知,比如一个线程等待某个条件成立,另一个线程在完成任务后通知等待的线程继续执行。

condition_variable 的基本原理

condition_variable 本身不保存状态,它只是一个等待和唤醒机制。线程通过调用 wait() 方法阻塞自己,直到其他线程调用 notify_one()notify_all() 来唤醒它。由于条件可能被虚假唤醒(spurious wakeup),所以等待时必须配合一个循环检查实际条件是否满足。

标准做法是使用 std::unique_lock<:mutex> 作为锁对象传入 wait 函数,因为 wait 会自动释放锁并在被唤醒后重新获取锁,避免死锁或竞争。

基本使用步骤

  • 定义一个 condition_variable 变量和一个 mutex 变量
  • 定义一个共享的条件标志(如 bool 值)来表示事件是否发生
  • 等待线程获取锁,然后在循环中调用 wait(),直到条件满足
  • 通知线程修改条件,释放锁后调用 notify_one() 或 notify_all()

代码示例:生产者-消费者模型

下面是一个简单的例子,演示两个线程如何通过 condition_variable 同步:

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <queue>

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::lock_guard<std::mutex>> lock(mtx);
        data_queue.push(i);
        std::cout << "Produced: " << i << std::endl;
        lock.unlock();
        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 << std::endl;
        }

        if (data_queue.empty() && finished) {
            std::cout << "Consumer exiting." << std::endl;
            break;
        }

        lock.unlock();
    }
}

int main() {
    std::thread c(consumer);
    std::thread p(producer);

    p.join();
    c.join();

    return 0;
}

关键点说明

wait() 的第二个参数是谓词(predicate),这是推荐写法。它等价于:

Reachout.ai Reachout.ai

一个AI驱动的视频开发平台,专为忙碌的企业家和销售团队打造

Reachout.ai 142 查看详情 Reachout.ai ```cpp while (!pred()) { wait(lock); } ```

这样可以防止虚假唤醒导致逻辑错误。

notify_one() 唤醒一个等待线程,适合一对一通信;notify_all() 唤醒所有等待线程,适用于广播场景,但可能带来性能开销。

注意:notify 操作不需要持有锁,但为了确保可见性,通常在修改共享数据并释放锁之后调用。

基本上就这些。合理使用 condition_variable 能有效协调线程执行顺序,避免忙等待,提升程序效率和响应性。

以上就是C++怎么使用条件变量condition_variable_C++线程同步与condition_variable使用的详细内容,更多请关注其它相关文章!


# 见性  # 网站推广的要点是什么  # 自助网站建设专业品牌  # 营销网站推广费用  # 金乡网站推广商机招标项目  # 专业网站推广哪家快  # 唐山seo外包报价  # 微网站建设公司北京  # 黑河seo快排  # 柳林网站推广有哪些公司  # 菜鸟新站seo优化  # 相关文章  # 并在  # 不需要  # c++线程同步  # 这是  # 如何使用  # 死锁  # 适用于  # 多线程  # 是一个  # 有锁  # red  # stream  # ios  # c++  # ai  # 工具 


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


相关推荐: 拼多多赚钱渠道_拼多多收益来源  Typer应用中动态命令行参数的解析与处理  Win11怎么修改默认浏览器_Windows 11设置Chrome为默认  React Router 嵌套组件中 URL 重定向问题的解决方案  C++如何比较两个字符串_C++ string compare函数与操作符对比  J*aScript实现单选按钮与关联输入框的联动禁用教程  在命令行怎么运行html项目_命令行运行html项目方法【教程】  如何在 Excel Online 和 Google 表格中更改日期格式  蛙漫移动版在线看 蛙漫手机浏览器直达入口  使用 Pandas 高效处理 .dat 文件:字符清理与数据计算  J*aScript实现动态背景色下的文本与按钮颜色自适应调整  AO3最新官网入口公告_2025AO3镜像站实时查询方法  精准捕获:如何在页面中监听除特定元素外的所有点击事件  c++20的std::jthread是什么_c++可中断线程与RAII式管理  押井守高度称赞《辐射4》:玩了八年都停不下来!  MinIO大规模对象列表性能瓶颈深度解析与外部元数据管理策略  学习通网页版快速入口 学习通官网网页版直接打开  如何在CSS中使用visited与link控制链接颜色_visited link伪类配合  AO3官方镜像站点汇总 AO3同人作品网页版直达链接  qq邮箱日历功能怎么用_创建日程与会议邀请的技巧  html5 app怎么运行环境_配html5 app运行环境【教程】  京东京造J1和网易云音乐氧气真无线有什么不同_国产电商蓝牙耳机音质对比  拼多多购物车商品数量无法修改如何处理 拼多多购物车操作优化方法  Golang如何使用new_Go new分配内存机制讲解  Win10如何开启蓝牙功能_Windows10找不到蓝牙开关解决方法  sublime怎么覆盖插件的默认快捷键_sublime快捷键优先级与设置  免费抖音短视频入口_抖音网页版短视频免费通道  Golang如何实现简单的Web表单_Golang表单提交与验证处理方法  Sublime Text怎么显示空格和制表符_Sublime显示不可见字符设置  Win11怎么隐藏桌面图标 Win11一键隐藏所有桌面元素及恢复显示  poki免费入口快捷访问 poki人气小游戏直接玩站点  BetterDiscord插件中安全更新用户简介的实践指南  QQ邮箱在线登录平台 QQ邮箱个人邮箱网页版入口  PS5 Pro有点优势但不多! 《燕云十六声》PS5平台与PC性能画面对比  C++ explicit关键字防止隐式转换_C++构造函数安全规范  文心一言怎样用批量生成做多版文案_文心一言用批量生成做多版文案【批量创作】  在J*a项目里如何构建对象之间的契约_接口约束的实际落地  Win11如何使用Windows Sandbox Win11沙盒功能开启与使用教程【详解】  Gmail邮箱申请注册直达_Gmail邮箱免费注册PC版官网入口2025  NetBeans Ant项目:自动化将资源文件复制到dist目录的教程  天猫2025双十一0点秒杀攻略 天猫爆款抢购时间  QQ邮箱官方登录入口_QQ邮箱网页版快捷使用平台  微信语音通话掉线如何解决 微信语音通话稳定优化方法  千牛数据看板网页版_千牛数据看板网页版访问方法  Django模型中自动计算可用余额的实现方法  vivo浏览器怎么扫描二维码 vivo浏览器内置扫一扫功能使用方法  搜狗浏览器如何使用密码生成器创建强密码 搜狗浏览器内置密码安全工具  汽水音乐在线解析 汽水音乐在线解析入口  Composer的 "check-platform-reqs" 命令有什么用_在部署前检查生产环境是否满足Composer依赖需求  知音漫客正版漫画平台_知音漫客官网账号登录 

搜索