Qt的互斥量用法

目的

互斥量的概念

互斥量是一个可以处于两态之一的变量:解锁和加锁。这样,只需要一个二进制位表示它,不过实际上,常常使用一个整型量,0表示解锁,而其他所有的值则表示加锁。互斥量使用两个过程。当一个线程(或进程)需要访问临界区时,它调用mutex_lock。如果该互斥量当前是解锁的(即临界区可用),此调用成功,调用线程可以自由进入该临界区。

另一方面,如果该互斥量已经加锁,调用线程被阻塞,直到在临界区中的线程完成并调用mutex_unlock。如果多个线程被阻塞在该互斥量上,将随机选择一个线程并允许它获得锁。

QMutex的用法

QMutex各QMutexLocker是基于互斥量的线程同步类,QMutex定义的实例是一个互斥量,QMutex主要提供了3个函数。

lock:

锁定互斥量,如果另外一个线程锁定了这个互斥量,它将阻塞执行直到其它线程解锁这个互斥量。

unlock:

解锁一个互斥量,需要与lock()配对使用。

tryLock():

试图锁定一个互斥量,如果成功锁定就返回true,如果其它线程已经锁定了这一个互斥量,就返回false,但不阻塞程序执行。

QMutexLocker:

是另外一个简化的互斥量处理的类。QMutexLocker的构造函数接受一个互斥量作为参数,将其锁定,QMutexLocker的析构函数则将此互斥量解锁。

例子

下面通过实现,两个程序对数量进行累加,演示互斥量的用法:

业务类:

cpp 复制代码
#include "business.h"
#include <QDebug>
#include <QThread>
#include <QDateTime>
int Business::s_num = 0;
QMutex Business::s_mutex;
Business::Business(QObject *parent) : QObject(parent)
{
}
int Business::getNum()
{
    return s_num;
}
void Business::addNum()
{
    //qDebug()<<"enter function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
    s_mutex.lock();
    s_num++;
    qDebug()<<"s_num="<<s_num;
    s_mutex.unlock();
    //qDebug()<<"exit function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
}
int Business::slot_getNum()
{
    return this->getNum();
}
void Business::slot_addNum()
{
    this->addNum();
}

线程类

cpp 复制代码
#include "businessInThread.h"
BusinessInThread::BusinessInThread(QObject *parent) : QObject(parent)
{
    m_pWorkerThread = new QThread();
    m_pBusiness = new Business();
    m_pBusiness->moveToThread(m_pWorkerThread);
    connect(m_pWorkerThread, &QThread::finished, m_pBusiness, &QObject::deleteLater);
    connect(this, &BusinessInThread::signal_getNum, m_pBusiness,
            &Business::slot_getNum, Qt::BlockingQueuedConnection);
    connect(this, &BusinessInThread::signal_addNum, m_pBusiness, &Business::slot_addNum);
    m_pWorkerThread->start();
}
BusinessInThread::~BusinessInThread()
{
    m_pWorkerThread->quit();
    m_pWorkerThread->wait();
}
int BusinessInThread::getNum()
{
    return signal_getNum();
}
void BusinessInThread::addNum()
{
     signal_addNum();
}

主程序

cpp 复制代码
#include <QCoreApplication>
#include "businessInThread.h"
int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);
    BusinessInThread businessInThread1;
    BusinessInThread businessInThread2;
    for(int i = 0; i < 5; i++)
    {
       businessInThread1.addNum();
       businessInThread2.addNum();
    }
    return a.exec();
}

运行结果:

如果去掉锁的话,就会这样:

QMutextLocker的简化用法

这个函数:

cpp 复制代码
void Business::addNum()
{
    //qDebug()<<"enter function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
    s_mutex.lock();
    s_num++;
    qDebug()<<"s_num="<<s_num;
    s_mutex.unlock();
    //qDebug()<<"exit function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
}

可以简化为:

cpp 复制代码
void Business::addNum()
{
    //qDebug()<<"enter function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
    //s_mutex.lock();
    QMutexLocker Locker(&s_mutex);
    s_num++;
    qDebug()<<"s_num="<<s_num;
    //s_mutex.unlock();
    //qDebug()<<"exit function addNum currentThreadId=" << QThread::currentThreadId()
    //       <<" currentTime="<<QDateTime::currentDateTime().toString("yyyy-MM-dd hh:mm:ss.zzz");
}

总结

互斥锁,就是只有锁定与解锁,两种状态,可以说,是最简单的锁,也是最实用的锁。

相关推荐
牙痛不能吃糖,哭2 分钟前
C++面试复习日记(8)2025.4.25,malloc,free和new,delete的区别
开发语言·c++
健康的猪6 分钟前
golang的cgo的一点小心得
开发语言·后端·golang
夜夜敲码27 分钟前
C语言教程(十六): C 语言字符串详解
c语言·开发语言
宋康34 分钟前
C语言结构体和union内存对齐
c语言·开发语言
꧁坚持很酷꧂40 分钟前
Linux Ubuntu18.04下安装Qt Craeator 5.12.9(图文详解)
linux·运维·qt
居然是阿宋44 分钟前
Kotlin高阶函数 vs Lambda表达式:关键区别与协作关系
android·开发语言·kotlin
ChoSeitaku1 小时前
17.QT-Qt窗口-工具栏|状态栏|浮动窗口|设置停靠位置|设置浮动属性|设置移动属性|拉伸系数|添加控件(C++)
c++·qt·命令模式
Cao1234567893211 小时前
简易学生成绩管理系统(C语言)
c语言·开发语言
The Future is mine1 小时前
C# new Bitmap(32043, 32043, PixelFormat.Format32bppArgb)报错:参数无效,如何将图像分块化处理?
开发语言·c#
亿坊电商1 小时前
PHP框架在微服务迁移中能发挥什么作用?
开发语言·微服务·php