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");
}

总结

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

相关推荐
一点媛艺1 小时前
Kotlin函数由易到难
开发语言·python·kotlin
姑苏风1 小时前
《Kotlin实战》-附录
android·开发语言·kotlin
奋斗的小花生2 小时前
c++ 多态性
开发语言·c++
魔道不误砍柴功2 小时前
Java 中如何巧妙应用 Function 让方法复用性更强
java·开发语言·python
闲晨2 小时前
C++ 继承:代码传承的魔法棒,开启奇幻编程之旅
java·c语言·开发语言·c++·经验分享
老猿讲编程2 小时前
一个例子来说明Ada语言的实时性支持
开发语言·ada
Chrikk3 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*3 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue3 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man3 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang