Qt 自定义event

本文章从属于 Qt实验室-CSDN博客系列

自定义事件携带数据

在某些情况下,可能需要通过事件携带一些额外的数据

可以通过自定义事件类来完成

一种实现方法是自定义一个QObject的子类,通过property来携带数据

复制代码
class MyEvent : public QObject,public QEvent
{
    Q_OBJECT
public:
    explicit MyEvent(QEvent::Type type,QObject *parent = nullptr);
}

这里通过一个自定义的窗口来发送自定义的事件,通过 QApplication::sendEvent(parent,evn);将事件传递给它的父窗口

复制代码
MyWidget::MyWidget(QWidget *parent)
    : QWidget{parent}
{
    QTimer* timer=new QTimer(this);
    timer->setInterval(1000);
    connect(timer,&QTimer::timeout,[=]{
        MyEvent* evn=new MyEvent(QEvent::Type(QEvent::User+101),this);
        evn->setProperty("x","1");
        evn->setProperty("y","2");
        //sendEvent阻塞式发送信号
        //发出去的信号需要接收者通过event()进行处理
        QApplication::sendEvent(parent,evn);
//        QApplication::postEvent(parent,evn);
    });
    timer->start();
}

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    this->resize(500,300);

    MyWidget* center=new MyWidget(this);
    m_center=center;

    this->setCentralWidget(center);
}

在父窗口中通过event()去处理事件,并且获得事件中携带的数据

复制代码
bool MainWindow::event(QEvent *event)
{
    if(event->type()==QEvent::Type(QEvent::User+101))
    {
        //需要dynamic_cast,qobject_cast会编译不过
//        MyEvent* eve=qobject_cast<MyEvent*>(event);
        MyEvent* eve=dynamic_cast<MyEvent*>(event);
        qDebug()<<eve->property("x")<<","<<eve->property("y");
        qDebug()<<"in event :"<<"...";
        return true;
    }
    return true;
}

sendEvent阻塞式与postEvent队列式

这里,QApplication::sendEvent(parent,evn)后,会立刻进入MainWindow::event()中去进行处理;

QApplication::postEvent(parent,evn);则是将事件放入队列后,立刻就返回了

复制代码
MyWidget::MyWidget(QWidget *parent)
    : QWidget{parent}
{
    QTimer* timer=new QTimer(this);
    timer->setInterval(1000);
    connect(timer,&QTimer::timeout,[=]{
        MyEvent* evn=new MyEvent(QEvent::Type(QEvent::User+101),this);
        evn->setProperty("x","1");
        evn->setProperty("y","2");
        //sendEvent阻塞式发送信号
        //发出去的信号需要接收者通过event()进行处理
//        QApplication::sendEvent(parent,evn);
        QApplication::postEvent(parent,evn);
        qDebug()<<"after send";

    });
    timer->start();
}

bool MainWindow::event(QEvent *event)
{

    if(event->type()==QEvent::Type(QEvent::User+101))
    {
        QThread::sleep(2);
        //需要dynamic_cast,qobject_cast会编译不过
//        MyEvent* eve=qobject_cast<MyEvent*>(event);
        MyEvent* eve=dynamic_cast<MyEvent*>(event);
//        QObject* eve=qobject_cast<QObject*>(event);
        qDebug()<<eve->property("x")<<","<<eve->property("y");
        qDebug()<<"in event :"<<"...";
        return true;
    }
    return false;
}

以上代码中,当使用 QApplication::sendEvent(parent,evn);时,打印的顺序是:

QVariant(QString, "1") , QVariant(QString, "2")

in event : ...

after send

当使用QApplication::postEvent(parent,evn);时,打印的顺序是:

after send

QVariant(QString, "1") , QVariant(QString, "2")

in event : ...

简单分析一下,当使用sendEvent发送事件时,会立刻进入到MainWindow::event(),从而打印QVariant(QString, "1") , QVariant(QString, "2")

in event : ...

从MainWindow::event()结束返回后,再回到QApplication::sendEvent()后面打印after send

体现了sendEvent的阻塞模式;

而使用postEvent发送事件时,会将事件放入队列,然后postEvent()返回了,于是打印了after send

等到事件队列处理了事件的时候,才会打印QVariant(QString, "1") , QVariant(QString, "2")

in event : ...

体现了postEvent的队列模式

但这里尽管发送事件的频率是1次/s,处理事件的频率是1次/2s,而使用postEvent发送事件时打印的after send和in event : ...依然是一一对应的。

这是因为发送事件和处理事件在同一个线程内,只有处理完了一个事件之后,timer才会继续下一个计时。

假如发送事件和处理事件不再同一个线程内,那么after send和in event : ...将不会一一对应。

相关推荐
侃侃_天下1 天前
最终的信号类
开发语言·c++·算法
echoarts1 天前
Rayon Rust中的数据并行库入门教程
开发语言·其他·算法·rust
Aomnitrix1 天前
知识管理新范式——cpolar+Wiki.js打造企业级分布式知识库
开发语言·javascript·分布式
每天回答3个问题1 天前
UE5C++编译遇到MSB3073
开发语言·c++·ue5
伍哥的传说1 天前
Vite Plugin PWA – 零配置构建现代渐进式Web应用
开发语言·前端·javascript·web app·pwa·service worker·workbox
小莞尔1 天前
【51单片机】【protues仿真】 基于51单片机八路抢答器系统
c语言·开发语言·单片机·嵌入式硬件·51单片机
我是菜鸟0713号1 天前
Qt 中 OPC UA 通讯实战
开发语言·qt
JCBP_1 天前
QT(4)
开发语言·汇编·c++·qt·算法
Brookty1 天前
【JavaEE】线程安全-内存可见性、指令全排序
java·开发语言·后端·java-ee·线程安全·内存可见性·指令重排序
百锦再1 天前
[特殊字符] Python在CentOS系统执行深度指南
开发语言·python·plotly·django·centos·virtualenv·pygame