Qt自定义事件实现及子线程向主线程传送事件消息

2024-06-11 20:58

本文主要是介绍Qt自定义事件实现及子线程向主线程传送事件消息,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

近期在又一次学习Qt的时候,由于要涉及到子线程与主线程传递消息,所以便琢磨了一下。顺便把有用的记录下来,方便自己以后查询及各位同仁的參考!

特此声明,本篇博文主要讲述有用的,也就是直接说明怎么实现,就不打算陈述一大堆理论啦,只是,还是建议大家去查查对应的理论比較好。这样能对Qt的消息传送机制的理解更加深入。

依据网上大多数人的资料,要实现自己定义消息,须要从QEvent 派生一个自己定义的事件;事实上也能够不须要,仅仅要使用QEvent::Type自己定义一个事件即可了。

在这里,本人把两种实现方法都在这里讲述一下!

 

一、这里先讲述使用 QEvent::Type 定义一个自己定义事件

      1、新建一个新的project “myEvent” 。在 ui 界面加入一个button,并未button加入 onclick() 响应函数。

      2、在 widget.h 头文件使用 QEvent::Type 定义两个自己定义事件。

      3、又一次实现 event() 虚函数

<span style="color:#000000">bool event(QEvent* e);</span>

      4、加入 #include<QEvent>

 

整个头文件例如以下:

 

<span style="color:#000000">#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QEvent>namespace Ui {
class Widget;
}class Widget : public QWidget
{Q_OBJECTpublic:explicit Widget(QWidget *parent = 0);~Widget();bool event(QEvent* e);private slots:void on_pushButton_clicked();private:Ui::Widget *ui;QEvent::Type myEvent1;QEvent::Type myEvent2;};#endif // WIDGET_H</span>

 

      5、使用QEvent::registerEventType() 静态函数为刚才两个自己定义事件注冊值。

      6、又一次实现 event()  函数 。

 

      7、在 button响应函数里面发送时间消息。

widget.cpp 文件例如以下:

 

<span style="color:#000000">#include "widget.h"
#include "ui_widget.h"
#include <QMessageBox>
#include <QCoreApplication>
#include <QDebug>Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);myEvent1 = static_cast<QEvent::Type>(QEvent::registerEventType(-1));myEvent2 = static_cast<QEvent::Type>(QEvent::registerEventType(-1));
}Widget::~Widget()
{delete ui;
}bool Widget::event(QEvent* e)
{if(e->type() == myEvent1){QMessageBox::warning(this, tr("event"), tr("myEvent1"), QMessageBox::Yes); return true;}else if(e->type() == myEvent2){QMessageBox::warning(this, tr("event"), tr("myEvent2"), QMessageBox::Yes); return true;}return QWidget::event(e);
}void Widget::on_pushButton_clicked()
{QCoreApplication::sendEvent(this, &QEvent(myEvent1));QCoreApplication::sendEvent(this, &QEvent(myEvent2));}</span>

编译执行后界面例如以下:

 

 

 

二、从 QEvent 派生一个自己定义事件类,类名取为 myEvent 。

 

      1、myevent.h 头文件例如以下。里面自己定义了三个自己定义事件,分别为 m_event1, m_event2, m_event3:

 

<span style="color:#000000">#ifndef MYEVENT_H
#define MYEVENT_H#include <QEvent>class myEvent : public QEvent
{
public:myEvent(Type e);public:static Type m_event1;static Type m_event2;static Type m_event3;
};#endif // MYEVENT_H
</span>

 

 

2 、在myevent.cpp 文件里面使用QEvent::registerEventType() 为自己定义的事件注冊。

myevent.cpp 文件例如以下:

 

<span style="color:#000000">#include "myevent.h"
#include <QEvent>QEvent::Type myEvent::m_event1 = static_cast<QEvent::Type>(QEvent::registerEventType());
QEvent::Type myEvent::m_event2 = static_cast<QEvent::Type>(QEvent::registerEventType());
QEvent::Type myEvent::m_event3 = static_cast<QEvent::Type>(QEvent::registerEventType());myEvent::myEvent(Type e):QEvent(e)
{
}
</span>

 

 

3、在 widget.cpp 文件 加入   myevent.h 头文件 。

4、改动 widget.cpp 文件中面的button响应函数例如以下:

 

<span style="color:#000000">void Widget::on_pushButton_clicked()
{myEvent e(myEvent::m_event1);QCoreApplication::sendEvent(this, &e);
}</span>

5、改动 widget.cpp 文件中面的 event() 函数例如以下:

 

 

<span style="color:#000000">bool Widget::event(QEvent* e)
{if(e->type() == myEvent1){QMessageBox::warning(this, tr("event"), tr("myEvent1"), QMessageBox::Yes); <pre name="code" class="cpp">        return true;}else if(e->type() == myEvent2){QMessageBox::warning(this, tr("event"), tr("myEvent2"), QMessageBox::Yes);
<pre name="code" class="cpp"><pre name="code" class="cpp">        return true;</span>

 

 

 }else if(e->type() == myEvent::m_event1){ QMessageBox::warning(this, tr("myEvent"), tr("m_event1"), QMessageBox::Yes); return true; } else if(e->type() == myEvent::m_event2){ QMessageBox::warning(this, tr("myEvent"), tr("m_event2"), QMessageBox::Yes); return true; } return QWidget::event(e);}

 
 

然后又一次编译执行,效果例如以下:

 

 

三 、 前面讲的都是在主线程里面传递事件消息,接下来讲述怎样 在子线程里面 往主线程 传递事件消息。

      1、从 QThread 派生一个自己定义事件类,类名取为 myThread , 并又一次实现 run() 虚函数。mythread.h 头文件例如以下:

 

<span style="color:#000000">#ifndef MYTHREAD_H
#define MYTHREAD_H#include <QThread>class myThread : public QThread
{Q_OBJECT
public:explicit myThread(QObject *parent = 0);signals:public slots:protected:void run();
};#endif // MYTHREAD_H</span>


      2、在mythread.cpp 里面又一次实现 run() 函数。在里面实现向主线程发送事件消息,mythread.cpp 文件例如以下:

 

 

<span style="color:#000000">#include "mythread.h"
#include "myevent.h"
#include <QCoreApplication>myThread::myThread(QObject *parent) :QThread(parent)
{
}void myThread::run()
{myEvent e(myEvent::m_event2);QCoreApplication::postEvent(this->parent(), new myEvent(myEvent::m_event2));//this->exec();
}
</span>


      3、在 widget.h        里面加入 mythread.h 头文件, 然后定义一个子线程对象, 例如以下:

 

#include"mythread.h"

 

myThread*m_pThread;

 

      4、在堆内存里面为m_pThread开辟一个内存空间,例如以下:

 

<span style="color:#000000"><span style="color:#000000;">Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);m_pThread = new myThread(this);。。</span>

 

。。。。 }</span>


      5、在 ui  界面 加入还有一个button。并为它加入 onclick() 事件响应。然后在里面执行子线程,例如以下:

 

 

<span style="color:#000000">void Widget::on_pushButton_2_clicked()
{m_pThread->start();
}
</span>

编译执行程序,效果例如以下:

 

 

四、QCoreApplication::postEvent(); 和 QCoreApplication::sendEvent();     的差别。

在前面的程序中发送 事件消息 的时候用到了 QCoreApplication::postEvent(); 和 QCoreApplication::sendEvent(); 两个函数,这里可不是随便使用的,这两个函数时又差别的!

 

      1、QCoreApplication::sendEvent();   依据Qt Asistant 里面的讲述。这个函数直接将事件消息直接发送给接受者进行处理,等到事件处理完成后才返回;而且使用它所传递的消息事件是在 栈(stack) 上创建的,也就是说它的内存空间是有编译器来自己主动管理的。

 

      2、QCoreApplication::postEvent();    依据Qt Asistant 里面的讲述,使用这个函数来传递时间消息时。它将事件消息发送到接受者的的消息队列里面,然后马上返回。不须要等到事件处理完成才返回;而且使用它所传递的消息事件是在 堆(heep) 上创建的。也就是说它的内存空间是又程序猿自己管理的,如用 new 创建的变量。

      这两个函数对事件的处理方式就像使用 repaint()  和 paint() 这两个函数对界面进行重画一样。前者直接对界面进行重画操作;后者先将重画事件放到消息队列里面,等到适当的时候在对界面进行重画操作。

 

       在上面的子线程给主线程传递消息的时候使用的就是QCoreApplication::postEvent();  函数。由于这里必须保证在子线程退出之前,若子线程所传递的事件消息还未被主线程处理的话,子线程所传递的消息仍然是可用的。

 

好了。主线程内事件的传递与子线程向主线程传递事件消息的方法就介绍到这里了。至于Qt 的事件传送机制。这里就没有怎么讲了。只是还是建议读者好好去了解一下的好。

这篇关于Qt自定义事件实现及子线程向主线程传送事件消息的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1052269

相关文章

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

禁止平板,iPad长按弹出默认菜单事件

通过监控按下抬起时间差来禁止弹出事件,把以下代码写在要禁止的页面的页面加载事件里面即可     var date;document.addEventListener('touchstart', event => {date = new Date().getTime();});document.addEventListener('touchend', event => {if (new

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P