QT中互斥锁QMutex的简单使用

意图
保护共享资源(对象、数据结构、代码段)
保证在同一时刻只有一个线程在使用共享资源
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,qt,开发语言)