本文主要是介绍QT6之多线程——子类化QObject和子类化QThread,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
备注:本文重点不是教怎么写多线程,这个官方示例依和网上示例一大把。
众所周知QT多线程有两种方法,一个是子类化QThread,另一个是子类化QObject。
Qt官方实际上是推荐后者,但实际运用中两者各有优劣和场景,并没有绝对的替代彼此。
多线程的必要性不言而喻,Qt默认的线程在Qt中称之为窗口线程,也叫主线程,负责窗口事件处理(鼠标事件、键盘事件等等)或者窗口控件数据的更新子线程负责后台的业务逻辑处理,子线程中不能对窗口对象做任何操作,这些事情需要交给窗口线程处理.
最常见的现象就是,当你新建项目后给QMainWindow或者QWidget对象的构造函数引入复杂耗时的逻辑时,然后再试着拖动窗口顶部结果卡死了。因为事件循环被阻塞了拖动窗口的事件执行不到了,然后就卡死了……
1、优劣对比
类别 | 优势 | 劣势 |
子类化QThread | 1、简单直观 2、控制权高 3、线程独立性 | 1、继承限制 2、信号槽连接不安全 |
子类化QObject | 1、灵活多样性 2、信号槽机制 默认支持事件循环 3、多继承支持 | 1、线程控制复杂 2、线程间通信需要额外处理 |
重点区别:继承QObject的多线程整个类都是在新的线程中,而继承QThread的多线程仅run函数内部在新的线程中 |
2、子类化QThread
QT虽然推崇后者但子类化QThread的方法截至最新版的Qt6仍然官方支持,在文档中也明确标注了两种实现方式,QThread仍然有很多人使用,我认为最重要的原因就在于它实现简单易于理解。
优:
- 简单直观: QThread是Qt提供的线程类,子类化QThread可以直接获得一个完整的线程对象,方便使用。
- 控制权高: 子类化QThread可以方便地控制线程的启动、停止、暂停、恢复等操作,封装了线程管理的相关方法。
- 线程独立性: 每个子类化QThread的实例都拥有自己独立的线程执行环境,不会受到其他线程的影响。
劣:
- 继承限制: 由于C++的单继承限制,当子类化QThread时,就不能再继承其他类了,这可能会限制代码的灵活性和复用性。
- 信号槽连接不安全: 在子类化QThread中使用信号槽连接时,需要注意线程安全的问题,因为信号槽连接是跨线程的,可能会引发竞态条件等问题。
子类化QThread实现一个线程如下就这么简单,退出线程时建议用注释了的标志位+锁的方式退出线程;
#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>
#include <QDebug>class MyThread : public QThread
{Q_OBJECT
public:explicit MyThread(QObject *parent = nullptr) : QThread(parent) {}// bool m_isCanRun;
protected:void run() override {qDebug() << "Thread started.";for (int i = 0; i < 100; ++i) {//m_isCanRun=false;//QMutexLocker locker(&m_lock);//if(!m_isCanRun)//在每次循环判断是否可以运行,如果不行就退出循环//{// return;// }QThread::sleep(1); // 模拟耗时操作qDebug() << "In thread:" << QThread::currentThreadId() << " - Count:" << i;}qDebug() << "Thread finished.";}
};#endif // MYTHREAD_H-----------------------------------------------------------------------------------------#include <QCoreApplication>
#include "MyThread.h"int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);MyThread thread;thread.start();
//stop时m_isCanRun置为truereturn a.exec();
}
QThread
只有run
函数是在新线程里的,其他所有函数都不在新线程里,看似简单但是也有弊端,假设要在一个子线程中处理多个任务,所有的处理逻辑都需要写到run()函数中,这样该函数中的处理逻辑就会变得非常混乱,不太容易维护 。
3、子类化QObject
优:
- 灵活多样: QObject是Qt中的基础类,它提供了丰富的功能和特性,可以用于创建各种不同类型的对象,包括线程对象。
- 信号槽机制: QObject提供了强大的信号槽机制,方便对象之间的通信,可以轻松实现跨线程的通信。
- 多继承支持: 由于QObject不受C++的单继承限制,因此可以在子类化QObject时继承其他类,提高了代码的灵活性和复用性。
劣:
- 线程控制复杂: 子类化QObject创建的对象并不直接拥有线程执行环境,因此在创建自定义线程时,需要额外管理线程的启动、停止等操作,相对较为复杂。
- 线程间通信需要额外处理: 子类化QObject创建的对象并不自带线程执行环境,因此在实现多线程时,需要额外处理线程的管理和通信,可能会增加一些复杂性。
4、选择建议
原则是“哪个熟练用哪个”,最合理的情况是这样:
- 如果只是创建一个简单的线程对象,执行一些任务,并且不需要与其他类进行继承关系,那么子类化QThread可能更为合适。
- 如果需要创建的对象既要拥有线程执行环境,又要拥有QObject的功能和特性,并且可能需要与其他类进行继承关系,那么子类化QObject可能更为适合。
5、示例
5.1 信号和槽机制通信无需互斥锁或信号量等同步机制
当需要与主线程进行通信时,可以通过信号和槽机制实现,而无需显式地使用互斥锁或信号量等同步机制,使得线程间通信更加简洁和安全。
假设在Worker对象的工作过程中,需要定期向主线程发送一些信息以更新界面。Worker对象可以定义一个信号,在需要发送信息时发射这个信号,而主线程可以连接这个信号到一个槽函数,以便在主线程中更新UI。
// Worker.h
#ifndef WORKER_H
#define WORKER_H#include <QObject>
#include <QThread>
#include <QDebug>class Worker : public QObject
{Q_OBJECTpublic:Worker() {}public slots:void doWork() {for (int i = 0; i < 5; i++) {qDebug() << "Working in thread" << QThread::currentThreadId();emit progressChanged(i); // 发射信号通知主线程工作进度QThread::sleep(1); // 模拟耗时任务}emit finished();}signals:void progressChanged(int value); // 声明一个信号,用于通知主线程工作进度void finished();
};#endif // WORKER_H
在主线程中,可以将这个信号连接到一个槽函数,以便在接收到信号时更新UI
// MainWindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QPushButton>
#include <QThread>
#include "Worker.h"class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr) : QMainWindow(parent) {QPushButton *button = new QPushButton("Start Thread", this);button->setGeometry(100, 100, 100, 30);Worker *worker = new Worker();QThread *thread = new QThread();worker->moveToThread(thread);connect(thread, &QThread::started, worker, &Worker::doWork);connect(worker, &Worker::finished, thread, &QThread::quit);connect(worker, &Worker::finished, worker, &QObject::deleteLater);connect(thread, &QThread::finished, thread, &QObject::deleteLater);// 将Worker对象的progressChanged信号连接到主线程的槽函数connect(worker, &Worker::progressChanged, this, &MainWindow::updateProgress);connect(button, &QPushButton::clicked, thread, &QThread::start);}public slots:void updateProgress(int value) {// 在这里更新主界面上的进度条或其他UI组件qDebug() << "Progress changed:" << value;}~MainWindow() {// 在主窗口销毁时,停止线程并等待线程结束thread->quit();thread->wait();}
};#endif // MAINWINDOW_H
5.2 线程默认自启动和销毁
1、启动:
在子类化QObject的线程中启动线程不是非得用信号和槽,默认start也可以启动。
2、重点是销毁:
在线程创建之后,这个的销毁是在主线程里进行,而是通过槽deleteLater进行安全的销毁。
而子类化QObject的线程中引入了两个对象,一个是QThread的对象(如下图thread),另一个是QObject的对象(如下图worker),它们的销毁有两种:
2.1 若QThread的对象在堆里即用指针或者说new实例化
QThread的对象必须通过槽函数deleteLater 安全销毁,并在析构函数中调用quit和wait;
2.2 若QThread的对象的存放在栈里
QThread的对象没必要调用槽函数deleteLater 安全销毁,但需在析构函数中调用quit和wait;
两种情况共性:QObject的对象一般都在堆里,直接调用槽函数deleteLater 安全销毁即可。
// MainWindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QThread>
#include "Worker.h"class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr) : QMainWindow(parent) {// 创建 Worker 对象和 QThread 对象worker = new Worker();thread = new QThread();// 将 Worker 对象移动到新的线程中worker->moveToThread(thread);// 连接信号和槽connect(thread, &QThread::started, worker, &Worker::doWork);connect(worker, &Worker::finished, thread, &QThread::quit);connect(worker, &Worker::finished, worker, &QObject::deleteLater);connect(thread, &QThread::finished, thread, &QObject::deleteLater);// 启动线程thread->start();}~MainWindow() {// 在主窗口销毁时,停止线程并等待线程结束thread->quit();thread->wait();}private:Worker *worker;QThread *thread;
};#endif // MAINWINDOW_H
5.3 多个Worker对象—多个线程管理
创建多个Worker对象并将它们移动到不同的线程中执行,建议使用QList指针管理。
注意对应的调用槽函数deleteLater 安全销毁,以及quit和wait。
// MainWindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QThread>
#include "Worker.h"class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr) : QMainWindow(parent) {// 创建三个 Worker 对象和对应的 QThread 对象for (int i = 0; i < 3; ++i) {Worker *worker = new Worker();QThread *thread = new QThread();// 将 Worker 对象移动到对应的线程中worker->moveToThread(thread);// 连接信号和槽connect(thread, &QThread::started, worker, &Worker::doWork);connect(worker, &Worker::finished, thread, &QThread::quit);connect(worker, &Worker::finished, worker, &QObject::deleteLater);connect(thread, &QThread::finished, thread, &QObject::deleteLater);// 启动线程thread->start();// 保存 Worker 对象和 QThread 对象workers.append(worker);threads.append(thread);}}~MainWindow() {// 在主窗口销毁时,停止所有线程并等待线程结束for (int i = 0; i < workers.size(); ++i) {threads[i]->quit();threads[i]->wait();}}private:QList<Worker*> workers;QList<QThread*> threads;
};#endif // MAINWINDOW_H
5.4 多个Worker对象—一个线程管理
不是每个Worker对象都必须对应一个独立的QThread对象。通常情况下,也可以将多个Worker对象移动到同一个QThread对象中执行,这样可以更好地利用系统资源,减少线程创建和销毁的开销。
请注意以下示例中调用thread的槽函数deleteLater安全销毁的位置,在for循环外部。
在Worker对象的finished()信号发出后,连接到QObject::deleteLater槽的代码应该在连接到thread对象的finished()信号之前,因为Worker对象在QThread结束时会被删除,如果先断开Worker对象的信号槽连接,再删除thread对象,会导致Worker对象的deleteLater操作无法执行,从而可能造成内存泄漏。
// MainWindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QThread>
#include "Worker.h"class MainWindow : public QMainWindow
{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr) : QMainWindow(parent) {// 创建一个 QThread 对象thread = new QThread();// 创建三个 Worker 对象for (int i = 0; i < 3; ++i) {Worker *worker = new Worker();// 将 Worker 对象移动到 QThread 对象中worker->moveToThread(thread);// 连接信号和槽connect(thread, &QThread::started, worker, &Worker::doWork);connect(worker, &Worker::finished, thread, &QThread::quit);connect(worker, &Worker::finished, worker, &QObject::deleteLater);// 保存 Worker 对象workers.append(worker);}// 连接 thread 对象的 finished() 信号// 在线程结束后,删除 thread 对象connect(thread, &QThread::finished, thread, &QObject::deleteLater);// 启动线程thread->start();}~MainWindow() {// 在主窗口销毁时,停止线程并等待线程结束thread->quit();thread->wait();}private:QList<Worker*> workers;QThread *thread;
};#endif // MAINWINDOW_H
总结:
子类化QObject实现多线程的方法,其实是QT优化QThread弊端后的一种更加灵活的封装,默认支持事件循环支持槽函数,整个类都在新的线程中,唯一注意的是它的销毁过程。
这篇关于QT6之多线程——子类化QObject和子类化QThread的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!