在此页面中,编写此示例代码是为了解释如何使用notify_one
:
#include <iostream>
#include <condition_variable>
#include <thread>
#include <chrono>
std::condition_variable cv;
std::mutex cv_m;
int i = 0;
bool done = false;
void waits()
{
std::unique_lock<std::mutex> lk(cv_m);
std::cout << "Waiting... \n";
cv.wait(lk, []{return i == 1;});
std::cout << "...finished waiting. i == 1\n";
done = true;
}
void signals()
{
std::this_thread::sleep_for(std::chrono::seconds(1));
std::cout << "Notifying...\n";
cv.notify_one();
std::unique_lock<std::mutex> lk(cv_m);
i = 1;
while (!done) {
lk.unlock();
std::this_thread::sleep_for(std::chrono::seconds(1));
lk.lock();
std::cerr << "Notifying again...\n";
cv.notify_one();
}
}
int main()
{
std::thread t1(waits), t2(signals);
t1.join(); t2.join();
}
然而,valgrind(实际上是 helgrind)抱怨说:
可能是竞争条件:条件变量 0x605420 已发出信号,但相关联的互斥锁 0x605460 未被信号线程锁定。
如果第二个线程在第一个线程之前运行并在其他线程之前到达cv.notify_one();
,它将向其他线程发出信号,而不会持有任何锁。
我实际上正在学习如何使用这些condition variables
并试图了解谁应该锁定/解锁与它们关联的互斥锁。所以我的问题是:这段代码做对了吗?或者是 helgrind 错了?