C++条件变量唤醒问题 & notify_one() 唤醒不及时问题

条件变量唤醒问题 & notify_one() 唤醒不及时问题

因为我对于 C++中条件变量的等待唤醒部分、notify_all & notify_one 的区别方面有些疑点,因此就有了以下的同 chatgpt 的沟通,希望同样能够帮助到大家

感叹于 chatgpt的强大

问题?

  • 我比较疑惑的是如果多个线程同时等待在一个条件变量上面(相同的判断函数),一个被唤醒了之后另一个会怎么样?还是继续阻塞?如果说获得互斥锁的那个线程没有更改条件(也就是说目前另一个线程的waiting的函数是能够返回true的)
  • notify_all 和 notify_one的区别?什么情况下用 notify_all()?为什么多个线程等待的话使用 notify_one()会出现效率问题?

答案

  • ❌(其实另一个线程还是会阻塞的,直到新的 notify 唤醒它)
    这里是错误的,改正后:如果是 notify_all(), 那么另外的线程同样也是早晚会被唤醒的,只是现在被阻塞了而已。当一个线程释放掉锁之后 其他的等待的线程还是会竞争锁的
  • 区别就是 noyify_all能够让所有的等待线程感知,所以能够直接选出来获得锁的,如果多个线程下使用 notify_one 可能会产生唤醒不够及时的问题,也就是他会随机唤醒一个,那 如果这个线程的条件是不满足的,那么这次唤醒就是失败了的。

请添加图片描述

请添加图片描述

请添加图片描述

请添加图片描述
请添加图片描述

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
notify_all 和 notify_one 都是 C++ 中的条件变量(condition variable)的成员函数,用于唤醒等待线程。 下面是一个简单的使用实例: ``` #include <iostream> #include <thread> #include <mutex> #include <condition_variable> std::mutex mtx; std::condition_variable cv; bool ready = false; void worker(int id) { std::unique_lock<std::mutex> lock(mtx); while (!ready) { std::cout << "Worker " << id << " is waiting..." << std::endl; cv.wait(lock); } std::cout << "Worker " << id << " is working..." << std::endl; } void notifier() { std::unique_lock<std::mutex> lock(mtx); ready = true; cv.notify_all(); } int main() { std::thread t1(worker, 1); std::thread t2(worker, 2); std::thread t3(worker, 3); std::this_thread::sleep_for(std::chrono::seconds(1)); std::thread t4(notifier); t1.join(); t2.join(); t3.join(); t4.join(); return 0; } ``` 上面的代码创建了三个 worker 线程和一个 notifier 线程。worker 线程会等待 ready 变量为 true,notifier 线程会将 ready 变量设置为 true,并通过 notify_all 函数唤醒所有等待线程。 运行上面的代码,输出结果如下: ``` Worker 1 is waiting... Worker 2 is waiting... Worker 3 is waiting... Worker 3 is working... Worker 2 is working... Worker 1 is working... ``` 可以看到,三个 worker 线程都等待在条件变量上,直到 notifier 线程唤醒它们,并且它们同时开始工作。如果将 notify_all 改为 notify_one,那么只会唤醒一个等待线程,输出结果会有所不同。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值