0

0

C++如何使用条件变量(condition_variable)?(线程同步)

穿越時空

穿越時空

发布时间:2025-12-29 17:24:03

|

977人浏览过

|

来源于php中文网

原创

条件变量是C++中实现线程等待-通知机制的核心工具,需与互斥锁配合使用,通过wait/notify配合共享条件变量解决线程同步问题,典型应用为生产者-消费者模型。

c++如何使用条件变量(condition_variable)?(线程同步)

条件变量(std::condition_variable)是 C++ 中实现线程间**等待-通知**机制的核心工具,常与互斥锁(std::mutex)配合使用,解决“某个线程需等待某条件成立后再继续执行”的问题。它本身不带状态,不能单独使用,必须搭配一个共享的布尔条件(通常用 std::atomic 或受保护的普通变量)。

基本用法:wait + notify_one/notify_all

核心流程是:等待线程先加锁,检查条件;若不满足,调用 wait() 原子地释放锁并挂起;另一线程修改条件后,调用 notify_one()notify_all() 唤醒等待者。

  • wait() 必须在已持有 mutex 的前提下调用,否则行为未定义
  • 推荐用带谓词的 wait 版本cv.wait(lock, []{ return condition; });,它自动处理“虚假唤醒”(spurious wakeup),避免手动循环检查
  • notify_one() 唤醒一个等待线程,notify_all() 唤醒所有,按需选择(如生产者-消费者中单个产品一般用 notify_one

典型场景:生产者-消费者模型

这是最常用的例子:一个或多个线程往队列里放数据(生产者),另一个或多个线程从中取数据(消费者)。队列为空时,消费者应等待;队列为满时,生产者应等待(若有限长)。

  • 共享数据(如 std::queue)必须由 std::mutex 保护
  • 空/满状态用条件变量分别控制:not_empty_cvnot_full_cv(或只用一个,靠条件判断)
  • 消费者等待:先 lock → 检查队列非空 → wait(not_empty_cv, 非空条件) → 取出元素 → unlock
  • 生产者通知:lock → 放入元素 → not_empty_cv.notify_one() → unlock

注意事项和常见陷阱

条件变量容易因细节出错,以下几点务必留意:

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

Z Code
Z Code

智谱AI推出的轻量级AI代码编辑器

下载
  • 永远不要对同一个 condition_variable 在不同 mutex 上 wait:每个 wait() 调用必须对应同一个 std::unique_lock<:mutex>
  • notify 应在修改共享状态后、释放锁前发出(即 notify 和状态更新要在同一临界区内),否则可能唤醒过早,导致等待线程再次检查失败
  • 避免在析构活跃的 condition_variable 前还有线程在 wait:确保所有等待线程已退出或被唤醒,否则程序可能崩溃
  • 不用 std::lock_guard 配合 wait():因为 wait() 需要能临时释放并重新获取锁,只能用 std::unique_lock

简单可运行示例(单生产者-单消费者)

以下代码片段展示基础用法(省略头文件和命名空间):

std::mutex mtx;
std::queue q;
std::condition_variable cv;
bool ready = false;

// 消费者线程
auto consumer = [&]{
    std::unique_lock lock(mtx);
    cv.wait(lock, [&]{ return !q.empty() || ready; });
    if (!q.empty()) {
        int val = q.front(); q.pop();
        std::cout << "Consumed: " << val << "\n";
    }
};

// 生产者线程
auto producer = [&]{
    std::this_thread::sleep_for(100ms);
    std::unique_lock lock(mtx);
    q.push(42);
    ready = true;
    cv.notify_one(); // 通知消费者
};

注意:真实项目中建议用 std::atomic_bool 替代普通 bool,并封装成线程安全队列类,避免裸写同步逻辑。

基本上就这些。条件变量不是万能锁,但它让线程协作更精确——不忙等、不抢锁、只在真正需要时才醒来。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

311

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

517

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

48

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

188

2025.08.29

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

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

466

2023.08.10

Golang 命令行工具(CLI)开发实战
Golang 命令行工具(CLI)开发实战

本专题系统讲解 Golang 在命令行工具(CLI)开发中的实战应用,内容涵盖参数解析、子命令设计、配置文件读取、日志输出、错误处理、跨平台编译以及常用CLI库(如 Cobra、Viper)的使用方法。通过完整案例,帮助学习者掌握 使用 Go 构建专业级命令行工具与开发辅助程序的能力。

1

2025.12.29

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

162

2025.12.26

压缩文件加密教程汇总
压缩文件加密教程汇总

本专题整合了压缩文件加密教程,阅读专题下面的文章了解更多详细教程。

52

2025.12.26

wifi无ip分配
wifi无ip分配

本专题整合了wifi无ip分配相关教程,阅读专题下面的文章了解更多详细教程。

108

2025.12.26

热门下载

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

精品课程

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

共58课时 | 3.1万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 2.9万人学习

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

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