意图
- 保护共享资源(对象、数据结构、代码段)
- 保证在同一时刻只有一个线程在使用共享资源
QMutex
// 阻塞加锁
void lock()
// 非阻塞加锁
bool tryLock(int timeout = 0)
// 解锁
void unlock()
QMutexLocker
- 对QMutex进行了RAII的封装
- 方便对锁的操作,避免忘记unlock
- 建议使用这种方式操作互斥锁
QMutex mutex;
QMutexLocker mutexLocker(&mutex);
使用例子
- 定义共享资源 AppData
// .h
class AppData {
public:
static int sharedNumber;
static QMutex sharedNumMutex;
};
// .cpp
int AppData::sharedNumber = 0;
QMutex AppData::sharedNumMutex;
- 启动线程
void Widget::on_pushButton_clicked()
{
m_thread2.start(); //先开启城线程2
m_thread1.start();
}
- 定义两个线程
1 不加锁
// thread1
void WorkThread1::run()
{
AppData::sharedNumber += 20;
AppData::sharedNumber -= 5;
qDebug() << "QThread1 ID:" << QThread::currentThreadId() << "result:" << AppData::sharedNumber;
}
// thread2
void WorkThread2::run()
{
AppData::sharedNumber += 2;
QThread::msleep(10); //延时一下,方便看出效果
// 不加锁,这里AppData::sharedNumber为线程1的执行结果和上面的 +2
// 即 +20-5+2=17
AppData::sharedNumber *= 10;
qDebug() << "QThread2 ID:" << QThread::currentThreadId() << "result:" << AppData::sharedNumber;
}
执行结果
QThread1 ID: 0x27a8 result: 17
QThread2 ID: 0x3028 result: 170
2 加锁
// thread1
void WorkThread1::run()
{
// 使用QMutexLocker加锁
QMutexLocker mutexLocker(&AppData::sharedNumMutex);
AppData::sharedNumber += 20;
AppData::sharedNumber -= 5;
qDebug() << "QThread1 ID:" << QThread::currentThreadId() << "result:" << AppData::sharedNumber;
}
// thread2
void WorkThread2::run()
{
// 使用QMutex加锁
AppData::sharedNumMutex.lock();
AppData::sharedNumber += 2;
QThread::msleep(10);
AppData::sharedNumber *= 10;
qDebug() << "QThread2 ID:" << QThread::currentThreadId() << "result:" << AppData::sharedNumber;
AppData::sharedNumMutex.unlock();
}
执行结果
QThread2 ID: 0x1314 result: 20
QThread1 ID: 0x1520 result: 35
线程2线先取到锁并加锁AppData::sharedNumMutex.lock(),所以线程1在执行到加锁操作时QMutexLocker mutexLocker(&AppData::sharedNumMutex),在这个位置会被阻塞住,只有等到线程2解锁,线程1才能加上锁并往下继续执行程序
学习资料
- QT帮助文档