我在解决获得锁定时出现的竞争条件方面遇到了一些麻烦。我有一个异步触发的大型计算。我需要先前的同步任务在大任务开始之前结束。大任务启动并等待条件变量,在理想情况下,小任务将在完成时通知条件变量。我的问题来自于太快启动的大任务,小任务触发一个尚未完全初始化的条件变量,因此实际上没有触发,导致程序被锁定。
我把它归结为这个最小的例子。
我认为这是一个常见问题。如何检查我的条件变量实际上是否已锁定互斥锁并被锁定?
#include <QCoreApplication>
#include <QObject>
#include <QFuture>
#include <QtConcurrent/QtConcurrent>
#include <QFutureWatcher>
#include <QWaitCondition>
class workUnit: public QObject
{
Q_OBJECT
public:
workUnit(QObject *parent)
{
m = new QMutex();
m->lock();
w=new QWaitCondition();
}
QWaitCondition* w;
QMutex* m;
public slots:
void runMe()
{
w->wait(m);
m->unlock();
//perform long computation
}
};
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
workUnit* mUnit=new workUnit(&a);
QFutureWatcher<void> w;
QObject::connect(&w, SIGNAL(finished()), &a, SLOT(quit()));
QFuture<void> f = QtConcurrent::run(mUnit,&workUnit::runMe);
w.setFuture(f);
_sleep(1000);//with this delay it works, without the delay it doesn't
mUnit->w->wakeAll();//This would be triggered by another process
return a.exec();
}
答案 0 :(得分:1)
QWaitCondition::wait
的文档指出:
必须首先通过调用线程锁定互斥锁。
您应该从构造函数中删除m->lock();
,并在调用runMe()
之前将其放在wait
函数中。