QT开发:详解 Qt 多线程编程核心类 QThread:基本概念与使用方法

1. 引言

        在现代应用程序开发中,多线程编程是一个关键技术,能够显著提高程序的效率和响应速度。Qt 是一个跨平台的 C++ 框架,其中 QThread 类是实现多线程编程的核心类。本文将深入详解 QThread 的基本概念、使用方法及其在实际应用中的重要性。

2. 基本概念

2.1 什么是 QThread?

        QThread 是 Qt 框架中的一个类,用于创建和管理线程。与标准库中的 std::thread 类似,QThread 提供了一种机制来执行并发任务,但它不仅限于此。QThread 还集成了 Qt 的信号和槽机制,使得线程间的通信更加方便和高效。

2.2 线程的生命周期

        一个线程的生命周期包括创建、启动、执行、结束几个阶段。QThread 类提供了一系列方法来控制和管理这些阶段:

  • start(): 启动线程,调用 run() 方法。
  • run(): 线程的工作函数,通常需要重载。
  • quit(): 让线程退出事件循环,但不终止线程。
  • terminate(): 强制终止线程,不推荐使用。
  • wait(): 等待线程结束。

3. 使用方法

3.1 继承 QThread 类

最常见的使用方式是通过继承 QThread 类并重载其 run() 方法。

3.1.1 示例代码

为了确保代码文件内容正确,请参考以下内容:

首先我们需要自定添加一个MyThread 类,会自动生成头文件和源文件,文件中代码会自动生成基础部分,其他需要手动编写,如下:

项目目录结构如下所示:

/my_project├── main.cpp├── mythread.h├── mythread.cpp└── my_project.pro

mythread.h

#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>
#include <QDebug>class MyThread : public QThread {Q_OBJECTpublic:MyThread() = default;~MyThread() = default;protected:void run() override {// 这里是线程的主要工作for (int i = 0; i < 5; ++i) {qDebug() << "Thread running:" << i;QThread::sleep(1);}}
};#endif // MYTHREAD_H

mythread.cpp

#include "mythread.h"
//这里仅做简单示例

 

main.cpp如下: 


int main() {MyThread thread;thread.start();thread.wait(); // 等待线程结束return 0;
}

运行结果如下:

 

3.1.2 知识精华

上面是一个简单 的 QThread 类应用示例,这里需要说明的一点是:

        在 Qt 项目中,.pro 文件是项目配置的重要部分,它不会自动生成,你需要手动创建它并添加相应的配置。如果你使用的是 Qt Creator 工具,它会自动为你生成一个初始的 .pro 文件,但你需要根据项目的需要手动修改和添加内容。

上面示例需要手动在.pro 文件中添加和修改配置项:

QT += core
QT -= guiCONFIG += console
CONFIG -= app_bundleTEMPLATE = appSOURCES += main.cpp \mythread.cppHEADERS += mythread.h

如果没有手动添加以上配置,Qt Creator工具中是无法构建运行的。当前示例.pro文件如下

3.2 使用 Worker-Object 模式

        继承 QThread 并重载 run() 方法虽然简单直观,但并不是最佳实践。更推荐的方式是使用 Worker-Object 模式,将工作对象移到新线程中。

#include <QObject>
#include <QThread>
#include <QDebug>// Worker类继承自QObject,用于执行线程中的工作任务
class Worker : public QObject {Q_OBJECTpublic slots:// 槽函数,执行工作任务void doWork() {// 模拟耗时操作,每次循环等待1秒for (int i = 0; i < 5; ++i) {qDebug() << "Worker running:" << i;QThread::sleep(1); // 线程休眠1秒}}
};int main() {// 创建一个QThread对象,用于管理新线程QThread thread;// 创建一个Worker对象,执行具体的工作任务Worker worker;// 将Worker对象移到新线程中运行worker.moveToThread(&thread);// 连接QThread的started信号和Worker的doWork槽// 当线程启动时,将调用Worker的doWork槽函数QObject::connect(&thread, &QThread::started, &worker, &Worker::doWork);// 连接Worker的doWork槽函数和QThread的quit槽// 当Worker完成工作后,将调用QThread的quit槽函数,退出线程事件循环QObject::connect(&worker, &Worker::doWork, &thread, &QThread::quit);// 连接QThread的finished信号和Worker的deleteLater槽// 当线程结束时,将调用Worker的deleteLater槽函数,删除Worker对象QObject::connect(&thread, &QThread::finished, &worker, &QObject::deleteLater);// 启动线程,开始执行工作任务thread.start();// 等待线程结束thread.wait();// 返回0,表示程序正常结束return 0;
}

Worker 类定义

  • Worker 类继承自 QObject
  • doWork 是一个槽函数,用于执行具体的工作任务。在这里,它模拟耗时操作,通过循环和线程休眠来演示。

 main 函数

  • 创建一个 QThread 对象 thread,用于管理新线程。
  • 创建一个 Worker 对象 worker,用于执行具体的工作任务。
  • 使用 moveToThread 方法将 Worker 对象移到新线程中运行。
  • 使用 QObject::connect 方法连接信号和槽:
    • 当线程启动时,调用 Worker 的 doWork 槽函数。
    • 当 Worker 完成工作后,调用 QThread 的 quit 槽函数,退出事件循环。
    • 当线程结束时,调用 Worker 的 deleteLater 槽函数,删除 Worker 对象。
  • 调用 thread.start() 启动线程,开始执行工作任务。
  • 调用 thread.wait() 等待线程结束。
  • 返回 0,表示程序正常结束。

4. 信号和槽

QThread 充分利用了 Qt 的信号和槽机制,使得线程间通信更加方便和安全。

#include <QObject>
#include <QThread>
#include <QDebug>// Worker类继承自QObject,用于执行线程中的工作任务
class Worker : public QObject {Q_OBJECTpublic slots:// 槽函数,执行工作任务void doWork() {// 这里是工作任务的代码emit workDone(); // 任务完成后发出信号}signals:// 信号,表示工作已经完成void workDone();
};int main() {// 创建一个QThread对象,用于管理新线程QThread thread;// 创建一个Worker对象,执行具体的工作任务Worker worker;// 将Worker对象移到新线程中运行worker.moveToThread(&thread);// 连接Worker的workDone信号和QThread的quit槽// 当Worker完成工作后,将调用QThread的quit槽函数,退出线程事件循环QObject::connect(&worker, &Worker::workDone, &thread, &QThread::quit);// 连接QThread的finished信号和Worker的deleteLater槽// 当线程结束时,将调用Worker的deleteLater槽函数,删除Worker对象QObject::connect(&thread, &QThread::finished, &worker, &QObject::deleteLater);// 启动线程,开始执行工作任务thread.start();// 在主线程中等待新线程结束thread.wait();// 返回0,表示程序正常结束return 0;
}

Worker 类定义

class Worker : public QObject {Q_OBJECTpublic slots:void doWork() {emit workDone(); // 任务完成后发出信号}signals:void workDone(); // 信号,表示工作已经完成
};
  • Worker 类继承自 QObject
  • doWork 是一个槽函数,用于执行具体的工作任务。在这里,它发出 workDone 信号,表示任务已经完成。
  • workDone 是一个信号,用于通知外部任务已经完成。

main 函数

int main() {QThread thread;Worker worker;worker.moveToThread(&thread);QObject::connect(&worker, &Worker::workDone, &thread, &QThread::quit);QObject::connect(&thread, &QThread::finished, &worker, &QObject::deleteLater);thread.start();thread.wait(); // 等待线程结束return 0;
}
  • 创建一个 QThread 对象 thread,用于管理新线程。
  • 创建一个 Worker 对象 worker,用于执行具体的工作任务。
  • 使用 moveToThread 方法将 Worker 对象移到新线程中运行。
  • 使用 QObject::connect 方法连接信号和槽:
    • 当 Worker 发出 workDone 信号时,调用 QThread 的 quit 槽函数,退出事件循环。
    • 当线程结束时,调用 Worker 的 deleteLater 槽函数,删除 Worker 对象。
  • 调用 thread.start() 启动线程,开始执行工作任务。
  • 调用 thread.wait() 在主线程中等待新线程结束。
  • 返回 0,表示程序正常结束。

5. 线程安全

        多线程编程中,线程安全是一个重要问题。QThread 提供了一些机制来帮助实现线程安全,例如 QMutex, QSemaphore, QWaitCondition 等。

#include <QMutex>
#include <QThread>
#include <QDebug>// 全局互斥锁,用于保护共享资源
QMutex mutex;// SafeWorker类继承自QObject,用于执行线程中的工作任务
class SafeWorker : public QObject {Q_OBJECTpublic slots:// 槽函数,执行工作任务void doWork() {// 锁定互斥锁,保护共享资源mutex.lock();for (int i = 0; i < 5; ++i) {qDebug() << "SafeWorker running:" << i;QThread::sleep(1); // 线程休眠1秒,模拟耗时操作}// 解锁互斥锁mutex.unlock();}
};int main() {// 创建一个QThread对象,用于管理新线程QThread thread;// 创建一个SafeWorker对象,执行具体的工作任务SafeWorker worker;// 将SafeWorker对象移到新线程中运行worker.moveToThread(&thread);// 连接QThread的started信号和SafeWorker的doWork槽// 当线程启动时,将调用SafeWorker的doWork槽函数QObject::connect(&thread, &QThread::started, &worker, &SafeWorker::doWork);// 连接SafeWorker的doWork槽函数和QThread的quit槽// 当SafeWorker完成工作后,将调用QThread的quit槽函数,退出线程事件循环QObject::connect(&worker, &SafeWorker::doWork, &thread, &QThread::quit);// 连接QThread的finished信号和SafeWorker的deleteLater槽// 当线程结束时,将调用SafeWorker的deleteLater槽函数,删除SafeWorker对象QObject::connect(&thread, &QThread::finished, &worker, &QObject::deleteLater);// 启动线程,开始执行工作任务thread.start();// 在主线程中等待新线程结束thread.wait();// 返回0,表示程序正常结束return 0;
}

全局互斥锁

QMutex mutex;

 创建一个全局的互斥锁 mutex,用于保护共享资源。

SafeWorker 类定义

class SafeWorker : public QObject {Q_OBJECTpublic slots:void doWork() {mutex.lock(); // 锁定互斥锁,保护共享资源for (int i = 0; i < 5; ++i) {qDebug() << "SafeWorker running:" << i;QThread::sleep(1); // 线程休眠1秒,模拟耗时操作}mutex.unlock(); // 解锁互斥锁}
};
  • SafeWorker 类继承自 QObject
  • doWork 是一个槽函数,用于执行具体的工作任务。在这里,通过锁定和解锁互斥锁来保护共享资源,确保线程安全。

main 函数

  • 创建一个 QThread 对象 thread,用于管理新线程。
  • 创建一个 SafeWorker 对象 worker,用于执行具体的工作任务。
  • 使用 moveToThread 方法将 SafeWorker 对象移到新线程中运行。
  • 使用 QObject::connect 方法连接信号和槽:
    • 当线程启动时,调用 SafeWorker 的 doWork 槽函数。
    • 当 SafeWorker 完成工作后,调用 QThread 的 quit 槽函数,退出事件循环。
    • 当线程结束时,调用 SafeWorker 的 deleteLater 槽函数,删除 SafeWorker 对象。
  • 调用 thread.start() 启动线程,开始执行工作任务。
  • 调用 thread.wait() 在主线程中等待新线程结束。
  • 返回 0,表示程序正常结束。

6. 实际应用中的重要性

        QThread 在实际应用中非常重要,特别是在需要进行耗时操作的场景下,如网络请求、数据库操作、大文件读写等。使用 QThread 可以显著提高应用程序的响应速度和用户体验。

7. 结论

        QThread 是 Qt 框架中一个强大且灵活的类,能够有效地实现多线程编程。通过本文的介绍,希望读者能够掌握 QThread 的基本概念、使用方法以及在实际应用中的重要性,从而更好地开发高性能的 Qt 应用程序。

8. 参考资料

  • Qt Documentation: QThread
  • Qt Documentation: Signals and Slots
  • Qt Documentation: Thread Support in Qt

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/433218.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对于 Vue CLI 项目如何引入Echarts以及动态获取数据

&#x1f680;个人主页&#xff1a;一颗小谷粒 &#x1f680;所属专栏&#xff1a;Web前端开发 很荣幸您能阅读我的文章&#xff0c;诚请评论指点&#xff0c;欢迎欢迎 ~ 目录 1、数据画卷—Echarts介绍 1.1 什么是Echarts&#xff1f; 1.2 Echarts官网地址 2、Vue CLI 项目…

第十三周:机器学习笔记

第十三周周报 摘要Abstract一、机器学习——Transformer&#xff08;上&#xff09;1. Sequence to Sequence(Seq 2 Seq&#xff0c;序列到序列模型) 的应用2. Transformer的结构2.1 Transformer encoder&#xff08;Transformer 编码器&#xff09; 二、Pytorch学习1. 网络模型…

将图片资源保存到服务器的盘符中

服务类 系统盘符&#xff1a;file-path.disk&#xff08;可能会变&#xff0c;配置配置文件dev中&#xff09;文件根路径&#xff1a;file-path.root-path&#xff08;可能会变&#xff0c;配置配置文件dev中&#xff09;http协议的Nginx的映射前缀&#xff1a;PrefixConstant.…

go解决引入私有包报错“Repository owner does not exist“的两种方式

当你写好引入的私有包,执行go mod tidy报错: Gogs: Repository owner does not exist fatal: Could not read from remote repository. Please make sure you have the correct access rights and the repository exists. 目前我的两种解决方案: 一、拉群整个…

freeRDP OPenssl

libusb需要下载 我使用的是VS2019编译 所以需要include 与vs2019 在cmake里面修改路径 C:/Users/JPM/source/repos/freeRDP/FreeRDP-stable-2.0/libusb-1.0.24/include/libusb-1.0 C:/Users/JPM/source/repos/freeRDP/FreeRDP-stable-2.0/libusb-1.0.24/VS2019/MS64/static/l…

模形式与态、势、感、知

模形式是数学中一个重要的研究领域&#xff0c;主要出现在数论、代数几何和表示论等多个学科中。模形式可以视为在某种意义上具有“对称性”的函数&#xff0c;这些函数在特定的条件下满足一定的变换性质。具体来说&#xff0c;模形式是定义在上半平面上的复值函数&#xff0c;…

第九节 Opencv自带颜色表操作

知识点&#xff1a;Look Up lTable&#xff08;LUT&#xff09;查找表 了解LUT查找表的作用与用法&#xff0c;代码实现与API介绍 -applyColorMap&#xff08;src,dst,COLORMAP&#xff09; -src表示输入图像 -dst表示输出图像 匹配到的颜色LUT&#xff0c;Opencv支持13种…

TDOA方法求二维坐标的MATLAB代码演示与讲解

引言 时间差定位(Time Difference of Arrival, TDOA)是一种用于确定信号源位置的技术,广泛应用于无线通信、声学定位等领域。通过测量信号到达多个接收器的时间差,可以计算出信号源的二维坐标。本文将通过MATLAB代码演示如何使用TDOA方法来求解二维坐标。 TDOA原理 TDOA…

第50篇 汇编语言实现中断<六>

Q&#xff1a;怎样设计汇编语言程序使用定时器中断实现实时时钟&#xff1f; A&#xff1a;此前我们曾使用轮询定时器I/O的方式实现实时时钟&#xff0c;而在本实验中将采用定时器中断的方式。新增的interval_timer.s间隔定时器的中断服务程序中增加了TIME变量&#xff0c;还更…

JavaScript的条件语句

if条件语句 if结构先判断一个表达式的布尔值&#xff0c;然后根据布尔值的真伪&#xff0c;执行不同的语句。所谓布尔值&#xff0c;指的是JavaScript 的两个特殊值&#xff0c;true表示真&#xff0c;false表示伪。 if语句语法规范 if(布尔值){语句;}var m3if(m3){console.l…

STM32嵌入式编程学习到提高:【4】UART串口打印

------------------------------------------------------------------------------------------------------------------------- 工程文件&#xff1a;放在百度云盘里&#xff0c;需要的自行下载&#xff01;&#xff01;&#xff01; 链接: https://pan.baidu.com/s/14gRne…

Flowable7.0.1框架严重bug,流程跳转到指定节点导致流程中断

一、Bug描述 使用7.0.1版本的 moveActivityIdsToSingleActivityId 或 moveExecutionsToSingleActivityId实现节点跳转&#xff0c;程序不会报错&#xff0c;但是act_ru_task 没有生成新的任务&#xff0c;导致流程中断&#xff0c;这是相当严重的bug。 经过多次测试&#xff…

【学习笔记】TLS/SSL握手之Records

TLS / SSL会话是由记录&#xff08;Records&#xff09;所组成&#xff0c;有4种records HandshakeAlertChange Cipher SpecApplication DataHandshake和Alert Records被分为子类型&#xff08;Subtypes&#xff09;&#xff1a; Handshake&#xff1a;Client HelloHandshake&a…

使用 Llama-index 实现的 Agentic RAG-Router Query Engine

前言 你是否也厌倦了我在博文中经常提到的老式 RAG(Retrieval Augmented Generation | 检索增强生成) 系统&#xff1f;反正我是对此感到厌倦了。但我们可以做一些有趣的事情&#xff0c;让它更上一层楼。接下来就跟我一起将 agents 概念引入传统的 RAG 工作流&#xff0c;重新…

Apache Iceberg 数据类型参考表

Apache Iceberg 概述-链接 Apache Iceberg 数据类型参考表 数据类型描述实例方法注意事项BOOLEAN布尔类型&#xff0c;表示真或假true, false用于条件判断&#xff0c;例如 WHERE is_active true。确保逻辑条件的正确性。INTEGER32位有符号整数42, -7可用于计算、聚合&#xf…

基于 Redis 实现滑动窗口的限流

⏳ 限流场景&#xff1a;突发流量&#xff0c;恶意流量&#xff0c;业务本身需要 基于 Redis 实现滑动窗口的限流是一种常见且高效的做法。Redis 是一种内存数据库&#xff0c;具有高性能和支持原子操作的特点&#xff0c;非常适合用来实现限流功能。下面是一个使用 Redis 实现…

Ubuntu环境下字体安装

本文介绍Ubuntu环境下字体安装。 软件&#xff08;如Qt应用软件&#xff09;开发过程中经常会涉及到字体的选择&#xff0c;有时候Ubuntu环境下并没有我们想要的字体&#xff0c;本文介绍常用字体及在Ubuntu环境下如何安装。 1.常用开源字体 有些字体商用并不是免费的&#…

Java搭建法律AI助手,快速实现RAG应用

使用AI4J快速接入RAG应用 | 结合Pinecone实现法律AI助手RAG应用 本博文给大家介绍一下如何使用AI4J快速接入OpenAI大模型&#xff0c;并且结合Pinecone向量数据库实现一个刑法AI助手的RAG应用。 介绍 由于SpringAI需要使用JDK17和Spring Boot3&#xff0c;但是目前很多应用依…

idea中.git文件夹存在但是没有git功能列表

1.问题&#xff1a; 该项目中已经将.git文件夹置入了&#xff0c;但是idea中却没有git相关的功能列表&#xff0c;如图&#xff1a; 2.解决办法&#xff1a; 在【文件】-【设置】-【版本控制】-【目录映射】中添加目录映射应用就好了 &#xff08;【File】 -> 【S…

为家增添幸福气息:八卦福·门牌的吉祥寓意

八卦福门牌作为一种独特的家居装饰&#xff0c;不仅美观&#xff0c;还蕴含着丰富的文化内涵和吉祥寓意。它通过融合八卦与“福”字的象征&#xff0c;提升家庭的幸福气息。以下将详细探讨八卦福门牌的吉祥寓意以及如何为家庭增添幸福。 八卦福门牌的吉祥寓意 八卦的象征&…