条件变量condition_variable用法学习[已迁移]

转自:https://www.cnblogs.com/fenghualong/p/13855360.html

1.介绍

 condition_variable类似于信号量机制,实现了线程的等待和唤醒。

  • wait() :阻塞等待的同时释放锁(原子操作),还可以添加阻塞判断函数,详见代码
  • notify_all() : 唤醒所有阻塞等待的线程
  • notify_one(): 唤醒某一个等待的线程

2.例子

锁+条件变量 实现等待唤醒。

#include <condition_variable>
#include <mutex>
#include <thread>

std::mutex mylock;
std::condition_variable condVar;

bool dataReady{false};

void waitingForWork() {
    std::cout << "Waiting ..." << std::endl;
    std::unique_lock<std::mutex> l(mylock);  // 加了unique_lock锁
    condVar.wait(l, []{return dataReady;});           // (4)
    std::cout << "Running ..." << std::endl;
}

void setDataReady() {
    {
        std::lock_guard<std::mutex> l{mylock};  // 加了lock_guard锁
        dataReady = true;
    }
    std::cout << "Data prepared, notify one" << std::endl;
    condVar.notify_one();                             // (3)
}

int main(int argc, char const *argv[])
{
    std::cout << "==========Begin==========" << std::endl;

    std::thread t1(waitingForWork);                    // (1)
    std::thread t2(setDataReady);                      // (2)

    t1.join();
    t2.join();

    std::cout << "===========End===========" << std::endl;
    cout<<endl;
    return 0;
}
// 运行结果
-> % ./main
==========Begin==========
Waiting ...
Data prepared, notify one
Running ...
===========End===========

如果不用锁和等待条件控制,可能会导致虚假唤醒和唤醒丢失问题。

2.1 wait函数原理

转自chatgpt

template<typename Predicate>
void wait(std::unique_lock<std::mutex>& lock, Predicate pred) {
    while (!pred()) {// 判断条件,条件返回false
        pthread_mutex_unlock(lock.mutex()->native_handle()); // 解锁unique_lock
        pthread_cond_wait(&cv, lock.mutex()->native_handle());// 等待唤醒
        pthread_mutex_lock(lock.mutex()->native_handle()); // 加锁
    }
}
  1. 外部锁定互斥量: 当调用 wait 函数时,互斥量已经被 std::unique_lock 锁定。std::unique_lock 提供了对互斥量的独占访问。
  2. 解锁互斥量并等待wait 函数会临时解锁互斥量,然后使当前线程进入阻塞状态,等待条件变量的通知。这一步骤是原子操作,确保在解锁和进入等待状态之间没有竞态条件。内部实现可能会调用互斥量的 unlock 方法和操作系统提供的等待机制。
  3. 被唤醒后重新锁定互斥量: 当条件变量被通知(通过 notify_onenotify_all),被阻塞的线程会被唤醒。在被唤醒后,wait 函数会重新锁定互斥量。 
  4. 检查条件并继续执行: 唤醒后,线程会检查条件是否满足。如果条件仍然不满足,线程会再次进入等待状态。否则,wait 函数返回,线程继续执行后续的代码。

3.用条件变量实现信号量

https://blog.csdn.net/qq_24447809/article/details/115906017

#pragma once
#include<mutex>
#include<condition_variable>
class semaphore {
public:
    semaphore(long count = 0) :count(count) {}
    void wait() {
        std::unique_lock<std::mutex>lock(mx);
        cond.wait(lock, [&]() {return count > 0; });
        --count;
    }
    void signal() {
        std::unique_lock<std::mutex>lock(mx);
        ++count;
        cond.notify_one();
    }

private:
    std::mutex mx;
    std::condition_variable cond;
    long count;
};

  

posted @ 2024-07-07 14:14  lypbendlf  阅读(16)  评论(0编辑  收藏  举报