当前位置: 首页 > article >正文

【C/C++】多线程开发:wait、sleep、yield全解析

文章目录

  • 多线程开发:wait、sleep、yield全解析
    • 1 What
      • 简要介绍
      • 详细介绍
        • `wait()` — 条件等待(用于线程同步)
        • `sleep()` — 睡觉,定时挂起
        • `yield()` — 自愿让出 CPU
    • 2 区别以及建议
      • 区别
      • 应用场景建议
    • 3 三者协作使用示例

多线程开发:wait、sleep、yield全解析

多线程开发中, wait()sleep()yield() 的正确使用对于控制线程行为、避免死锁、提升性能至关重要。


1 What

简要介绍

方法属于哪类是否释放锁作用
wait()同步原语(条件变量)✅ 释放锁当前线程等待某个条件满足,挂起执行。
sleep()时间控制❌ 不释放锁当前线程强制睡眠一段时间。
yield()调度建议❌ 不释放锁当前线程主动让出 CPU,允许其他线程执行。

详细介绍

wait() — 条件等待(用于线程同步)
  • std::condition_variable::wait()(C++11)
  • Java 的 Object.wait() 也类似

行为:

  • 线程在等待某个条件成立
  • 释放所持的互斥锁(mutex),进入阻塞状态。
  • 条件满足后,被 notify_one() / notify_all() 唤醒,重新竞争锁并继续执行。

C++ 示例:

std::mutex mtx;
std::condition_variable cv;
bool ready = false;void worker() {std::unique_lock<std::mutex> lock(mtx);cv.wait(lock, []{ return ready; }); // 释放锁等待条件成立std::cout << "Condition met, working..." << std::endl;
}

sleep() — 睡觉,定时挂起
  • std::this_thread::sleep_for()
  • std::this_thread::sleep_until()

行为:

  • 当前线程被强制阻塞指定时间
  • 不会释放任何锁
  • 时间一到,线程进入就绪状态(可被调度)。

示例:

std::mutex mtx;void worker() {std::lock_guard<std::mutex> lock(mtx);std::this_thread::sleep_for(std::chrono::seconds(2));std::cout << "Wake up after 2 seconds.\n";
}

🔺 注意:如果你在持锁状态下调用 sleep(),会让其它线程“饿死”或引发死锁。


yield() — 自愿让出 CPU
  • std::this_thread::yield()(C++11)
  • 实际为对操作系统 sched_yield() 的封装

行为:

  • 当前线程主动放弃 CPU 时间片
  • 线程回到就绪队列,允许其他同优先级线程执行。
  • 不会阻塞、不休眠、不释放锁。

示例:

void busy_wait() {while (!ready) {std::this_thread::yield(); // 避免 CPU 忙等}
}

2 区别以及建议

区别

特性wait()sleep()yield()
阻塞线程?✅ 是✅ 是❌ 否(可能暂停)
是否释放锁?✅ 是(必须配合 mutex 使用)❌ 否❌ 否
唤醒条件notify_one() / notify_all()时间到被 OS 再次调度
典型用途线程间同步,条件等待限制频率、延时模拟忙等时避免 CPU 占用
所属 APIcondition_variable / pthread_condstd::this_thread::sleep_for()std::this_thread::yield()

应用场景建议

场景推荐方法
等待某个状态改变wait()
模拟延时 / 限速 / 轮询间隔sleep()
忙等 / 自旋等待中降低 CPU 占用yield()

3 三者协作使用示例

验证目标:

  • 主线程 sleep 控制节奏
  • 子线程 yield 等待任务
  • 条件变量 wait 同步工作

代码目标:

  • 主线程每 1 秒产生一个任务。
  • 工作线程使用 yield() 自旋检查是否有任务。
  • 一旦任务准备好,工作线程使用 wait() 等待条件变量通知,再执行任务。
#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <atomic>
#include <chrono>std::mutex mtx;
std::condition_variable cv;
std::atomic<bool> ready{false};
int task_counter = 0;// 工作线程:等待任务并执行
void worker_thread() {while (true) {// ⏳ 自旋检查任务是否准备好(减少 wait 的频率)while (!ready.load()) {std::this_thread::yield(); // 主动让出 CPU}std::unique_lock<std::mutex> lock(mtx);cv.wait(lock, [] { return ready.load(); }); // 等待通知,并释放锁// 模拟任务处理std::cout << "[Worker] Executing task #" << task_counter << std::endl;ready = false;  // 标记任务完成if (task_counter >= 5) break;  // 结束条件}
}// 主线程:周期性产生任务
void task_producer() {for (int i = 1; i <= 5; ++i) {std::this_thread::sleep_for(std::chrono::seconds(1)); // 模拟任务生成延迟{std::lock_guard<std::mutex> lock(mtx);task_counter = i;ready = true;std::cout << "[Main] Task #" << i << " ready\n";}cv.notify_one(); // 通知工作线程}
}int main() {std::thread worker(worker_thread);std::thread producer(task_producer);producer.join();worker.join();std::cout << "All tasks finished.\n";return 0;
}

输出:

[Main] Task #1 ready
[Worker] Executing task #1
[Main] Task #2 ready
[Worker] Executing task #2
[Main] Task #3 ready
[Worker] Executing task #3
[Main] Task #4 ready
[Worker] Executing task #4
[Main] Task #5 ready
[Worker] Executing task #5
All tasks finished.

代码行为解析:

  • 主线程:

    • 每秒产生一个任务,调用 sleep_for()
    • 使用锁保护 task_counterready 状态。
    • 调用 cv.notify_one() 唤醒工作线程。
  • 工作线程:

    • 使用 yield() 忙等(在任务未准备好前避免 CPU 占用)。
    • 使用 condition_variable::wait() 挂起,等待任务准备。
    • 被通知后执行任务,打印日志,重置状态。
http://www.lryc.cn/news/2385388.html

相关文章:

  • uint8_t是什么数据类型?
  • SystemUtils:你的Java系统“探照灯“——让环境探测不再盲人摸象
  • 对象存储(Minio)使用
  • yolov11使用记录(训练自己的数据集)
  • 历史数据分析——宁波港
  • 知识宇宙:技术文档该如何写?
  • DeepSeek 赋能数字农业:从智慧种植到产业升级的全链条革新
  • 技嘉主板怎么开启vt虚拟化功能_技嘉主板开启vt虚拟化教程(附intel和amd开启方法)
  • Java 并发编程高级技巧:CyclicBarrier、CountDownLatch 和 Semaphore 的高级应用
  • PT5F2307触摸A/D型8-Bit MCU
  • 矩阵方程$Ax=b$的初步理解.
  • 线性代数中的向量与矩阵:AI大模型的数学基石
  • [特殊字符] 使用增量同步+MQ机制将用户数据同步到Elasticsearch
  • LeetCode 2942.查找包含给定字符的单词:使用库函数完成
  • 【mediasoup】MS_DEBUG_DEV 等日志形式转PLOG输出
  • 打卡第27天:函数的定义与参数
  • python训练营day34
  • 人工智能在医疗影像诊断上的最新成果:更精准地识别疾病
  • 塔能节能平板灯:点亮苏州某零售工厂节能之路
  • 3DMAX插件UV工具UV Tools命令参数详解
  • Docker 与微服务架构:从单体应用到容器化微服务的迁移实践
  • 《岁月深处的童真》
  • 文件夹图像批处理教程
  • RL电路的响应
  • 30-消息队列
  • 跨域解决方案之JSONP
  • 【AI测试革命】第七期:AI性能测试的深度实践——从智能建模到自动化调优的全链路升级
  • Thinkphp6使用token+Validate验证防止表单重复提交
  • AppAgentx 开源AI手机操控使用分享
  • Axure设计之带分页的穿梭框原型