Qt中在子线程中刷新UI的方法

Qt中在子线程中刷新UI的方法

在Qt中UI界面并不是线程安全的,意味着在子线程中不能随意操作UI界面组件(比如按钮、标签)等,如果强行操作这些组件有可能会导致程序崩溃。那么在Qt中如何在子线程中刷新UI控件呢?

两种方法:

方法一:使用信号槽机制。

第一步:创建一个QWidget项目,并且在其中添加一个继承自QThread的子线程类QWorkThread,如下:

c++ 复制代码
#ifndef QWORKTHREAD_H
#define QWORKTHREAD_H

#include <QObject>
#include<QThread>


class QWorkThread : public QThread
{
    Q_OBJECT
public:
    explicit QWorkThread(QThread *parent = nullptr);
    QWorkThread(QWidget* pWidget,QThread *parent = nullptr);
    QWidget* m_pWidget = nullptr;

protected:
    virtual void run() override;

signals:
    void UpdateUI(QString strInfo);
};

#endif // QWORKTHREAD_H

说明:

1.自定义一个信号UpdateUI用来刷新主线程UI。

2.重写run函数,使用emit 发射UpdateUI信号如下:

c++ 复制代码
void QWorkThread::run()
{
    emit UpdateUI("刷新控件");
}

第二步:在MainWindow中定义一个槽函数,用来响应UpdateUI信号:

c++ 复制代码
#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>

QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

public slots:
    void UpdateWidgetUI(QString strInfo);


private:
    Ui::MainWindow *ui;
};
#endif // MAINWINDOW_H

第三步:调用connect连接UpdateUI信号和UpdateWidgetUI槽函数,刷新UI:

C++ 复制代码
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "qworkthread.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    QWorkThread* pThread = new QWorkThread(ui->label);
    connect(pThread,&QWorkThread::UpdateUI,this,&MainWindow::UpdateWidgetUI);
    connect(pThread,&QThread::finished, pThread, &QThread::deleteLater);
    pThread->start();
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::UpdateWidgetUI(QString strInfo)
{
    ui->label->setText(strInfo);
}

程序运行前后结果:

方法二:使用invokeMethod()方法。

第一步:增加一个带QWidget的构造函数,并且在QWorkThread定义一个QWidget 类型的变量用来保存要刷新的控件。如上QWorkThread头文件中。

第二步:修改QWorkThread::run方法如下:

c++ 复制代码
void QWorkThread::run()
{
    if(m_pWidget != nullptr)
      QMetaObject::invokeMethod(m_pWidget,"setText",Qt::QueuedConnection,Q_ARG(QString,"刷新控件2"));
}

其中调用QMetaObject::invokeMethod方法。

程序运行结果如下:

虽然Qt不允许子线程直接操作UI界面,但通过信号与槽机制或invokeMethod()方法,子线程和UI线程可以安全地进行交互。这些机制确保了多线程程序的稳定性,避免了UI更新时可能出现的线程安全问题。

参考文章:Qt/C++面试【速通笔记五】---子线程与GUI线程安全交互

相关推荐
LYOBOYI1235 小时前
qml练习:创建地图玩家并且实现人物移动(2)
开发语言·qt
得鹿梦鱼、7 小时前
Qt/halcon 总结halcon字典数据用法和QDataSteam序列化与反序列化结构体实例
qt·tuple字典·qdatasteam·序列化结构体·反序列化结构体
世转神风-8 小时前
qt-基础打印-不换行打印
开发语言·qt
老歌老听老掉牙8 小时前
PyQt5中RadioButton互斥选择的实现方法
开发语言·python·qt
864记忆10 小时前
Qt Creator 常用命令的中英文对照表
开发语言·qt
Larry_Yanan11 小时前
Qt多进程(六)共享内存和信号量
开发语言·qt
东方忘忧12 小时前
Qt使用QDesktopServices::openUrl打开系统默认应用(如浏览器,文件,文件夹和邮件)
开发语言·qt
计算机内卷的N天12 小时前
qt的模态和非模态状态
开发语言·qt
charlie071 天前
qmake工程不能显示手动配置的Kit
qt
深蓝海拓1 天前
PySide6从0开始学习的笔记(二十) qdarkstyle的深入应用
笔记·python·qt·学习·pyqt