【QT5 多线程示例】互斥锁

news/2025/3/26 2:25:48/

互斥锁

互斥锁介绍:【C++并发编程】(三)互斥锁:std::mutex。原理都一样,这里就不赘述了。

QMutex 是 Qt 框架中提供的一个互斥锁类,主要包括以下成员函数:

  • lock():试图锁定互斥量。如果另一个线程已经锁定了这个互斥量,调用线程将被阻塞,直到那个线程解锁。
  • unlock():解锁互斥量,允许其他线程锁定它。
  • tryLock():尝试锁定互斥量,不阻塞调用线程。如果互斥量被锁定,函数返回 false;如果成功锁定,返回 true
  • tryLock(int timeout):尝试在指定的毫秒数内锁定互斥量。如果超时仍未锁定,返回 false;如果成功锁定,返回 true
  • locked():查询互斥量当前是否被锁定。如果被锁定,返回 true;否则返回 false

在简单的函数中,可以直接使用 QMutex 的 lock()unlock() 成员函数。但在复杂的函数中,使用 QMutexLocker 自动管理互斥锁更为安全和方便。QMutexLocker 与C++标准中std::lock_guard的用法差不多,而且也是基于 RAII(Resource Acquisition Is Initialization)机制的,在构造时自动锁定互斥量,在析构时自动解锁。

下面给出示例代码:
https://github.com/BinaryAI-1024/QtStudy/tree/master/thread/mutex

//myworker.h
#ifndef MYWORKER_H
#define MYWORKER_H#include <QObject>
#include <QMutex>
#include <QDebug>class MyWorker : public QObject
{Q_OBJECT
signals:void finished();public:explicit MyWorker(QObject *parent = nullptr);// 获取 counter 的值static int getCounter();public slots:void doWork(int id);private:static QMutex mutex;static int counter;
};#endif // MYWORKER_H
//myworker.cpp
#include "myworker.h"// 静态成员初始化
QMutex MyWorker::mutex;
int MyWorker::counter = 0;MyWorker::MyWorker(QObject *parent): QObject(parent)
{
}void MyWorker::doWork(int id)
{qDebug() << "Worker" << id << "started work.";for (int i = 0; i < 100000; ++i) {QMutexLocker locker(&mutex); // // 加锁以保护数据++counter;}emit finished(); // 发送完成信号
}int MyWorker::getCounter()
{return counter;
}
//main.cpp
#include <QCoreApplication>
#include <QThread>
#include "myworker.h"
#include <QTimer>
#include <QMutex>
#include <QMutexLocker>int main(int argc, char *argv[])
{QCoreApplication app(argc, argv);QMutex mutex;                  // 创建一个互斥锁const int numThreads = 3;      // 定义要启动的线程数量QThread *threads[numThreads];  // 创建一个 QThread 指针数组,用于存储线程对象MyWorker *workers[numThreads]; // 创建一个 MyWorker 指针数组,用于存储工作对象int finishedCount = 0;         // 初始化一个计数器,用于跟踪完成的线程数量// 循环创建和启动线程for (int i = 0; i < numThreads; ++i) {threads[i] = new QThread; // 创建一个新线程workers[i] = new MyWorker; // 创建一个新工作对象workers[i]->moveToThread(threads[i]); // 将工作对象移动到新线程中// 连接线程的 started 信号到工作对象的 doWork 槽,使用 QTimer 确保在事件循环开始后执行QObject::connect(threads[i], &QThread::started, workers[i], [=]() {QTimer::singleShot(0, workers[i], [=]() { workers[i]->doWork(i); });});// 连接工作对象的 finished 信号到线程的 quit 槽,以便任务完成后退出线程的事件循环QObject::connect(workers[i], &MyWorker::finished, threads[i], &QThread::quit);// 连接工作对象的 finished 信号到工作对象的 deleteLater 槽,以便任务完成后删除工作对象QObject::connect(workers[i], &MyWorker::finished, workers[i], &QObject::deleteLater);// 连接线程的 finished 信号到线程的 deleteLater 槽,以便线程退出后删除线程对象QObject::connect(threads[i], &QThread::finished, threads[i], &QObject::deleteLater);// 连接工作对象的 finished 信号到一个 lambda 表达式,用于更新计数器并检查所有线程是否完成QObject::connect(workers[i], &MyWorker::finished, [&finishedCount, &mutex, i]() {QMutexLocker locker(&mutex); // 加锁以保护计数器的访问++finishedCount; // 增加已完成线程的计数qDebug() << "Worker" << i << "finished. " ;if (finishedCount == numThreads) { // 检查是否所有线程都已完成// 每个线程使counter增加100000,正确结果应该是:numThreads*100000qDebug() << "counter:" << MyWorker::getCounter();}});threads[i]->start(); // 启动线程}return app.exec();
}

结果:

Worker 1 started work.
Worker 0 started work.
Worker 2 started work.
Worker 1 finished. 
Worker 2 finished. 
Worker 0 finished. 
counter: 300000

互斥锁在代码中保护了main.cpp中的 ++finishedCount;以及mythread.cpp中的++counter;,避免了多个线程同时执行这些操作导致的错误。QMutexLocker能够自动管理局部作用域内互斥锁的加锁和开锁。

另外,在这段代码中,不需要调用 thread.wait() 是因为 Qt 的信号和槽机制自动管理了线程的生命周期。通过 QThread::quit()QObject::deleteLater(),线程在任务完成后自动退出并清理资源。


http://www.ppmy.cn/news/1583108.html

相关文章

2、pytest核心功能(进阶用法)

目录 1、标记&#xff08;Markers&#xff09;&#xff1a; 自定义插件 内置标记 2、夹具&#xff08;Fixtures&#xff09;&#xff1a; 夹具得用法 夹具作用域 3、钩子&#xff08;hook&#xff09;&#xff1a; 这篇是最重要的 测试文件中需要用到的 总的来说 有以下…

【论文笔记】生成对抗网络 GAN

GAN 2014 年&#xff0c;Ian Goodfellow 等人提出生成对抗网络&#xff08;Generative Adversarial Networks&#xff09;&#xff0c;GAN 的出现是划时代的&#xff0c;虽然目前主流的图像/视频生成模型是扩散模型&#xff08;Diffusion Models&#xff09;的天下&#xff0c…

HarmonyOS Next~鸿蒙AI功能开发:Core Speech Kit与Core Vision Kit的技术解析与实践

HarmonyOS Next&#xff5e;鸿蒙AI功能开发&#xff1a;Core Speech Kit与Core Vision Kit的技术解析与实践 一、鸿蒙AI功能开发的生态定位与核心能力 在鸿蒙操作系统&#xff08;HarmonyOS&#xff09;的生态布局中&#xff0c;AI功能开发是提升用户体验与设备智能化的核心方…

debian12运行sql server2022(docker):导入.MDF .LDF文件到容器

过程大纲 docker run在基础配置之上增加挂载信息 修改文件权限&#xff0c;确保所有用户有rw权限 进入docker交互命令行 登录数据库 执行数据库EXE命令导入数据库文件数据 docker run在基础配置之上增加挂载信息 docker run -d \-v /home/ying/Downloads/StuXk:/var/opt/mssql…

关于 Redis 缓存一致

为了提升系统性能&#xff0c;常常会引入 Redis 作为缓存。数据通常会存储在持久化的数据源&#xff08;如 MySQL 数据库&#xff09;中&#xff0c;同时在 Redis 中保存一份副本。当数据源中的数据发生变化时&#xff0c;如果不能及时同步到 Redis 缓存&#xff0c;或者缓存中…

Netty源码—3.Reactor线程模型四

大纲 5.NioEventLoop的执行总体框架 6.Reactor线程执行一次事件轮询 7.Reactor线程处理产生IO事件的Channel 8.Reactor线程处理任务队列之添加任务 9.Reactor线程处理任务队列之执行任务 10.NioEventLoop总结 8.Reactor线程处理任务队列之添加任务 (1)Reactor线程执行一…

带你了解Java无锁并发CAS

带你了解Java无锁并发CAS 在多核处理器时代&#xff0c;并发编程已成为提升系统性能的核心手段。传统的同步机制&#xff08;如synchronized和ReentrantLock&#xff09;通过互斥锁实现线程安全&#xff0c;但其存在以下关键问题&#xff1a; 性能损耗&#xff1a;线程阻塞/唤…

常见中间件漏洞攻略-Tomcat篇

一、 CVE-2017-12615-Tomcat put方法任意文件写入漏洞 第一步&#xff1a;开启靶场 第二步&#xff1a;在首页抓取数据包&#xff0c;并发送到重放器 第三步&#xff1a;先上传尝试一个1.txt进行测试 第四步&#xff1a;上传后门程序 第五步&#xff1a;使用哥斯拉连接 二、后…