C++11中完善了信号量与条件变量(参见:C++11中线程及信号量与条件变量),但是condition_variable
的一些实现是存在问题的,特别是在Windows下会很容易出现假唤醒的情形(VS2015的实现,在Win10下非常容易出现,而Win7下偶尔出现)。
为避免此种假唤醒情形,需要使用带Predicate
参数的重载接口,以下是Event接口的模拟实现:
class XuEvent{ // No signal when init
public:
void wait(){
std::unique_lock<std::mutex> lker(m_mtx);
m_bNotified = false;
m_cv.wait(lker, [this] {return this->m_bNotified; });
}
bool wait(int nSec, int nMillSec=0){
std::unique_lock<std::mutex> lker(m_mtx);
m_bNotified = false;
std::chrono::seconds secs(nSec);
std::chrono::milliseconds mills(nMillSec);
auto ret = m_cv.wait_for(lker, secs + mills, [this] {return this->m_bNotified; });
//return (ret != std::cv_status::timeout);
return ret;
}
void notifyOne(){
m_bNotified = true;
m_cv.notify_one();
}
void notifyAll(){
m_bNotified = true;
m_cv.notify_all();
}
private:
bool m_bNotified = false;
std::mutex m_mtx;
std::condition_variable m_cv;
};