qt-C++笔记之事件过滤器

qt-C++笔记之事件过滤器

—— 杭州 2024-02-25


在这里插入图片描述

code review!

文章目录

  • qt-C++笔记之事件过滤器
    • 一.使用事件过滤器和不使用事件过滤器对比
      • 1.1.使用事件过滤器
      • 1.2.不使用事件过滤器
      • 1.3.比较
    • 二.Qt 中事件过滤器存在的意义
    • 三.为什么要重写QObject的eventFilter方法?使用QObject默认的eventFilter方法不可以吗?
    • 四.事件过滤器的使用场景
      • 4.1. 灵活的事件处理:
      • 4.2. 代码解耦:
      • 4.3. 条件事件处理:
      • 4.4. 监控和调试:
      • 4.5. 创建可重用的行为:
    • 五.使用事件过滤器和不使用事件过滤器优缺点
      • 使用事件过滤器
      • 不使用事件过滤器
    • 六.《Qt6 C++开发指南》——事件过滤器部分
    • 七.《Qt程序设计基础 基于银河麒麟桌面操作系统》

一.使用事件过滤器和不使用事件过滤器对比

在Qt框架中,事件过滤器通常通过重写QObject类的eventFilter()方法来实现。下面提供两个简单的示例,展示在Qt C++中使用和不使用事件过滤器的情况:

1.1.使用事件过滤器

首先,创建一个新的类(例如MyFilter),从QObject派生,并重写eventFilter()方法来拦截和处理特定的事件。

#include <QObject>
#include <QEvent>
#include <QDebug>class MyFilter : public QObject {Q_OBJECTprotected:bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::KeyPress) {qDebug() << "A key press event has been filtered.";// 处理事件return true;  // 返回true表示事件已被处理,不需要进一步传递}// 其他事件传递给基类的eventFilter方法return QObject::eventFilter(obj, event);}
};

然后,你需要在你想要过滤其事件的对象上安装这个过滤器:

#include <QApplication>
#include <QWidget>int main(int argc, char *argv[]) {QApplication a(argc, argv);QWidget window;MyFilter *filter = new MyFilter(&window);window.installEventFilter(filter);  // 安装事件过滤器window.show();return a.exec();
}

1.2.不使用事件过滤器

如果不使用事件过滤器,你可能会直接在对象的事件处理方法中处理所有的事件,例如在QWidget的子类中重写keyPressEvent()方法:

#include <QWidget>
#include <QKeyEvent>
#include <QDebug>class MyWidget : public QWidget {Q_OBJECTprotected:void keyPressEvent(QKeyEvent *event) override {qDebug() << "A key press event has been received.";// 处理事件QWidget::keyPressEvent(event);  // 调用基类的keyPressEvent方法}
};int main(int argc, char *argv[]) {QApplication a(argc, argv);MyWidget window;window.show();return a.exec();
}

在这个例子中,MyWidget会接收并处理所有的按键事件,而不是通过一个单独的过滤器来进行筛选。

1.3.比较

使用事件过滤器的方式更加灵活,因为你可以在不修改原有类的情况下,通过安装或移除过滤器来动态改变事件处理行为。此外,过滤器可以用于多个对象,不需要为每个对象编写重复的事件处理代码。

而不使用事件过滤器的方式更加直接,通常用于处理特定对象的事件。这种方式能够保证事件不会被其他对象意外截获,但是如果需要在多个对象之间共享相同的事件处理逻辑,可能会导致代码重复。

二.Qt 中事件过滤器存在的意义

Qt框架中的事件过滤器(Event Filter)是一个强大的功能,它允许对象拦截和处理发给其他对象的事件。事件过滤器的存在意义主要体现在以下几个方面:

  1. 灵活的事件处理:在Qt中,通常情况下,事件是被发送到目标对象的事件处理函数中(如mousePressEventkeyPressEvent等)。通过安装事件过滤器,你可以在事件到达目标对象之前对其进行拦截和处理,这允许你在不修改目标对象代码的情况下增加额外的行为。

  2. 代码解耦:事件过滤器可以帮助你减少类之间的耦合。例如,可以在一个单独的类中处理所有相关的事件,而不是将这些处理逻辑分散到多个子类中。

  3. 条件事件处理:你可以根据需要决定是否要处理某个事件,或者在某些条件满足时改变事件的行为。事件过滤器可以基于动态条件(如应用程序的状态)来决定是否和如何处理事件。

  4. 监控和调试:事件过滤器可以用来监控事件流,帮助开发者理解事件是如何在应用程序中流转的,这对于调试事件相关的问题非常有用。

  5. 创建可重用的行为:通过事件过滤器,可以创建可以附加到多个对象上的通用行为,而无需每个对象都实现相同的事件处理代码。

要使用事件过滤器,你需要重写QObjecteventFilter方法,并使用installEventFilter方法将过滤器对象安装到目标对象上。当事件发生并且传递给目标对象之前,eventFilter方法会被调用,你可以在这个方法中决定如何处理事件。

下面是一个简单的例子,说明如何为一个对象安装事件过滤器:

class MyEventFilter : public QObject
{
protected:bool eventFilter(QObject *obj, QEvent *event) override{if (event->type() == QEvent::KeyPress) {// 处理按键事件// ...return true; // 返回true表示事件被处理,不再传递}// 将事件传递给基类的事件过滤器return QObject::eventFilter(obj, event);}
};// 在某个地方使用事件过滤器
MyEventFilter *filter = new MyEventFilter();
targetObject->installEventFilter(filter);

在这个例子中,MyEventFilter类重写了eventFilter方法来处理按键事件。当目标对象targetObject接收到按键事件时,eventFilter方法会被调用,我们可以在这里决定如何处理该事件。如果我们返回true,则表示事件已经被处理并且不应该继续传递;如果返回false,事件将会继续传递给目标对象的相应事件处理方法。

三.为什么要重写QObject的eventFilter方法?使用QObject默认的eventFilter方法不可以吗?

在Qt中,QObject类提供了一个默认的实现eventFilter方法,但这个默认实现只是简单地返回false,表示它没有处理该事件,事件应该继续传递给目标对象的事件处理方法。换句话说,如果你不重写eventFilter方法,那么安装事件过滤器将没有任何实际效果,因为所有事件都会按照正常的流程继续传递。

重写eventFilter方法的目的是为了在事件到达目标对象之前介入处理流程。你可以在这个方法中编写自定义的逻辑来决定是否拦截某个事件,改变事件的默认行为,或者在事件传递到目标对象之前进行一些特殊处理。

下面是QObjecteventFilter方法默认实现的伪代码表示:

bool QObject::eventFilter(QObject *watched, QEvent *event)
{// 默认实现,不做任何处理,返回false表示不拦截事件return false;
}

当你创建一个新的类继承自QObject或者其子类,并且你希望该类能够以某种方式处理或者监控其他对象的事件时,你就需要提供eventFilter的自定义实现。例如,你可能想要记录所有的鼠标点击事件,或者在用户按下特定快捷键时触发某个动作。这时,你就需要重写eventFilter方法,如下所示:

class MyCustomFilter : public QObject
{// ...protected:bool eventFilter(QObject *watched, QEvent *event) override{if (event->type() == QEvent::MouseButtonPress) {// 处理鼠标按下事件// ...return true; // 事件被处理,不再传递}// 对于不处理的事件,调用基类的eventFilter方法return QObject::eventFilter(watched, event);}
};

在这个例子中,MyCustomFilter类重写了eventFilter方法以便在鼠标按下时执行一些自定义操作。如果eventFilter方法返回true,则表示事件已经被处理,不应该继续传递;如果返回false,则事件会按照正常的事件处理流程继续传递给目标对象。

四.事件过滤器的使用场景

当然可以,下面是针对每个场景的main.cpp文件的完整示例。

4.1. 灵活的事件处理:

#include <QApplication>
#include <QLineEdit>
#include <QDebug>
#include <QObject>class HelpEventFilter : public QObject {
protected:bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::KeyPress) {QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);if (keyEvent->key() == Qt::Key_F1) {qDebug() << "Help requested for:" << obj;return true;}}return QObject::eventFilter(obj, event);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);QLineEdit lineEdit;HelpEventFilter helpFilter;lineEdit.installEventFilter(&helpFilter);lineEdit.show();return app.exec();
}

4.2. 代码解耦:

#include <QApplication>
#include <QWidget>
#include <QDebug>
#include <QObject>class UpdateEventFilter : public QObject {
protected:bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::MouseButtonPress) {qDebug() << "Updating components due to click on:" << obj;return true;}return QObject::eventFilter(obj, event);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);QWidget widget;UpdateEventFilter updateFilter;widget.installEventFilter(&updateFilter);widget.show();return app.exec();
}

4.3. 条件事件处理:

#include <QApplication>
#include <QWidget>
#include <QDebug>
#include <QObject>class GameEventFilter : public QObject {bool gameRunning;public:GameEventFilter(bool running) : gameRunning(running) {}protected:bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::KeyPress && gameRunning) {qDebug() << "Game is running - event processed for:" << obj;return true;}return QObject::eventFilter(obj, event);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);QWidget widget;GameEventFilter gameFilter(true);widget.installEventFilter(&gameFilter);widget.show();return app.exec();
}

4.4. 监控和调试:

#include <QApplication>
#include <QWidget>
#include <QDebug>
#include <QObject>class DebugEventFilter : public QObject {
protected:bool eventFilter(QObject *obj, QEvent *event) override {qDebug() << "Event:" << event->type() << "for object:" << obj;return QObject::eventFilter(obj, event);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);QWidget widget;DebugEventFilter debugFilter;widget.installEventFilter(&debugFilter);widget.show();return app.exec();
}

4.5. 创建可重用的行为:

#include <QApplication>
#include <QWidget>
#include <QDebug>
#include <QObject>
#include <QDragEnterEvent>
#include <QMimeData>class DragDropEventFilter : public QObject {
protected:bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::DragEnter) {QDragEnterEvent *dragEvent = static_cast<QDragEnterEvent *>(event);if (dragEvent->mimeData()->hasFormat("text/plain")) {qDebug() << "Drag enter event for:" << obj;dragEvent->accept();return true;}}return QObject::eventFilter(obj, event);}
};int main(int argc, char *argv[]) {QApplication app(argc, argv);QWidget widget;widget.setAcceptDrops(true);DragDropEventFilter dragDropFilter;widget.installEventFilter(&dragDropFilter);widget.show();return app.exec();
}

在每个例子中,我们创建了一个应用程序实例和一个主窗口部件,然后创建了一个事件过滤器并将其安装到主窗口部件上。在实际应用中,你可能需要调整过滤器逻辑以适应你的具体需求。这些示例中的eventFilter函数的返回值很重要,如果它返回true,则表明事件已被处理,不会再传递给对象;如果返回false,则事件会继续传递给对象的event方法。

五.使用事件过滤器和不使用事件过滤器优缺点

在计算机编程和系统设计中,事件过滤器是一种用来决定是否应该处理或忽略某个事件的机制。事件可以是用户操作、系统生成的消息或者是由硬件设备产生的信号等。使用事件过滤器和不使用事件过滤器可以带来不同的效果和影响,下面列举了一些对比点:

使用事件过滤器

优点:

  1. 效率提升:当事件数量很大时,过滤器可以帮助减少无关事件的处理,从而节省CPU时间和内存资源。
  2. 更好的用户体验:过滤器可以确保只有相关和重要的事件被处理,避免了不必要的干扰,提高应用程序的响应性。
  3. 可维护性:过滤器可以帮助组织和管理事件处理逻辑,使代码更加清晰,易于维护和更新。
  4. 灵活性:过滤器可以动态地添加或移除,方便地调整哪些事件应该被处理。
  5. 安全性:过滤掉不必要的或潜在危险的事件,可以减少安全风险。

缺点:

  1. 复杂性:设计和实现事件过滤器会增加系统的复杂度。
  2. 性能开销:虽然过滤器可以提高整体效率,但是过滤器本身的运行也会消耗一定的性能。
  3. 可能的错误过滤:如果过滤器逻辑设计不当,可能会错误地过滤掉应该处理的事件。

不使用事件过滤器

优点:

  1. 简单性:不需要考虑过滤逻辑,实现起来比较简单直接。
  2. 无额外性能开销:没有过滤器运行的开销,每个事件都直接送进事件处理流程。
  3. 完整性:所有事件都会被送至处理程序,没有遗漏的风险。

缺点:

  1. 效率低下:没有过滤机制意味着所有事件,无论是否相关,都会被处理,这可能会导致不必要的性能负担。
  2. 用户体验不佳:处理大量无关事件可能会导致应用程序响应变慢,影响用户体验。
  3. 可维护性差:随着事件处理逻辑的增多,代码可能变得难以管理和维护。
  4. 安全风险:处理所有事件可能会使系统面临更多的安全威胁,因为不加选择的处理可能包含恶意的事件。

总结来说,使用事件过滤器可以帮助提高系统效率,提升用户体验,加强安全性,并有助于代码维护。然而,它也会带来额外的复杂性和性能开销。不使用事件过滤器则在实现上更为简单,但可能会导致效率低下和维护困难等问题。在实际应用中,是否使用事件过滤器取决于具体的需求、资源限制和安全考虑。

六.《Qt6 C++开发指南》——事件过滤器部分

在这里插入图片描述

七.《Qt程序设计基础 基于银河麒麟桌面操作系统》

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

groovy:XmlParser 读 Freeplane.mm文件,生成测试案例.csv文件

Freeplane 是一款基于 Java 的开源软件&#xff0c;继承 Freemind 的思维导图工具软件&#xff0c;它扩展了知识管理功能&#xff0c;在 Freemind 上增加了一些额外的功能&#xff0c;比如数学公式、节点属性面板等。 强大的节点功能&#xff0c;不仅仅节点的种类很多&#xff…

时序预测 | Matlab实现基于GRNN广义回归神经网络的光伏功率预测模型

文章目录 效果一览文章概述源码设计参考资料效果一览 文章概述 1.时序预测 | Matlab实现基于GRNN广义回归神经网络的光伏功率预测模型 2.单变量时间序列预测; 3.多指标评价,评价指标包括:R2、MAE、MBE等,代码质量极高; 4.excel数据,方便替换,运行环境2020及以上。 广义回…

python 运算符总结

什么是运算符 什么是运算符? 先看如下示例 549 例子中&#xff0c;4 和 5 被称为操作数&#xff0c; 称为运算符。 而Python 语言支持以下类型的运算符: 算术运算符比较&#xff08;关系&#xff09;运算符赋值运算符逻辑运算符位运算符成员运算符身份运算符运算符优先级 …

OPENSSL-PKCS7入门知识介绍

1 PKCS7数据结构说明 p7包括6种数据内容&#xff1a;数据(data),签名数据&#xff08;sign&#xff09;&#xff0c;数字信封数据&#xff08;enveloped&#xff09;&#xff0c;签名数字信封数据&#xff08;signed_and_enveloped&#xff09;&#xff0c;摘要数据&#xff08…

软件测试过程中如何有效的开展接口自动化测试

一.简介 接口自动化测试是指使用自动化测试工具和脚本对软件系统中的接口进行测试的过程。其目的是在软件开发过程中&#xff0c;通过对接口的自动化测试来提高测试效率和测试质量&#xff0c;减少人工测试的工作量和测试成本&#xff0c;并且能够快速发现和修复接口错误&…

自己测试CSDN质量分3

你好你好你好你好你好你好你好你好你好你好你好你好你好你好你好你好你好 质量分测试网址

智慧校园|智慧校园管理小程序|基于微信小程序的智慧校园管理系统设计与实现(源码+数据库+文档)

智慧校园管理小程序目录 目录 基于微信小程序的智慧校园管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、微信小程序前台 2、管理员后台 &#xff08;1&#xff09;学生信息管理 &#xff08;2&#xff09; 作业信息管理 &#xff08;3&#xff09;公告…

蓝桥杯算法赛 第 6 场 小白入门赛 解题报告 | 珂学家 | 简单场 + 元宵节日快乐

前言 整体评价 因为适逢元宵节&#xff0c;所以这场以娱乐为主。 A. 元宵节快乐 题型: 签到 节日快乐&#xff0c;出题人也说出来自己的心愿, 祝大家AK快乐! import java.util.Scanner;public class Main {public static void main(String[] args) {System.out.println(&qu…

栈和堆什么意思,Rust所有权机制又是什么

栈和堆什么意思 栈&#xff1a;存储基本数据类型和引用数据类型的指针引用(地址)&#xff0c;基本数据类型占据固定大小的内存空间。 堆&#xff1a;存储引用数据类型的值&#xff0c;引用数据类型包括对象&#xff0c;数组和函数&#xff0c;在堆中&#xff0c;引用数据类型…

AI论文速读 | 【综述】(LLM4TS)大语言模型用于时间序列

题目&#xff1a;Large Language Models for Time Series: A Survey 作者&#xff1a;Xiyuan Zhang , Ranak Roy Chowdhury , Rajesh K. Gupta and Jingbo Shang 机构&#xff1a;加州大学圣地亚哥分校&#xff08;UCSD&#xff09; 网址&#xff1a;https://arxiv.org/abs/…

JAVA工程师面试专题-《Redis》篇

目录 一、基础 1、Redis 是什么 2、说一下你对redis的理解 3、Redis 为什么这么快&#xff1f; 4、项目中如何使用缓存&#xff1f; 5、为什么使用缓存&#xff1f; 6、Redis key 和value 可以存储最大值分别多是多少&#xff1f; 7、Redis和memcache有什么区别&#xf…

Folx Pro Mac中文p破解版如何使用?为您带来Folx Pro 详细使用教程!

​ Folx pro 5 中文版是mac上一款功能强大的老牌加速下载软件&#xff0c;新版本的Folx pro整体界面非常的简洁和漂亮&#xff0c;具有非常好用的分类管理功能&#xff0c;支持高速下载、定时下载、速度控制、iTunes集成等功能。Folx pro兼容主流的浏览器&#xff0c;不但可以下…

开源世界的学术问题

自由软件基金会是1983年成立的&#xff0c;到现在是41年。正好很有意思的是&#xff0c;在去年还有一篇文章&#xff08;CSDN 的翻译&#xff09;&#xff0c;专门在质疑说成立 40 年的自由软件基金会是不是已经快不行了&#xff0c;所以我们会用这个标题叫做兴衰发展历程来介绍…

Excel的中高级用法

单元格格式&#xff0c;根据数值的正负分配不同的颜色和↑ ↓ 根据数值正负分配颜色 2-7 [蓝色]#,##0;[红色]-#,##0 分配颜色的基础上&#xff0c;根据正负加↑和↓ 2↑-7↓ 其实就是在上面颜色的代码基础上加个 向上的符号↑&#xff0c;或向下的符号↓ [蓝色]#,##0↑;[红色…

uni-app vue3 setup nvue中webview层级覆盖问题

核心就是这两行&#xff0c;&#x1f923;发现设置后不能点击了&#xff0c;这个玩意可能只能弹窗打开的时候动态的修改 position: static, zindex: 0onLoad(options > {loadWebview()})function loadWebview() {let pageInfo uni.getSystemInfoSync();width.value pageI…

ROS 2的前世今生 | ROS 2学习笔记

自2015年底首次踏入ROS&#xff08;Robot Operating System&#xff09;的世界以来&#xff0c;我在机器人领域的旅程已近九年。这段历程始于团队几位志同道合的朋友在业余时间的自发学习&#xff0c;逐渐演变成成立了一个致力于英特尔硬件平台与ROS框架集成优化的专业团队&…

力扣--动态规划1014.最佳观光组合

思路分析: 初始化左侧景点的评分为第一个景点的评分&#xff0c;最终结果为0。从第二个景点开始遍历数组。对于每个景点&#xff0c;计算当前观光组合的得分&#xff0c;即当前景点的评分 左侧景点的评分 - 两者之间的距离。更新最终结果为当前得分和之前结果的较大值。更新左…

matlab simulink变压器温度仿真

1、内容简介 略 48-可以交流、咨询、答疑 2、内容说明 略 3、仿真分析 略 matlab simulink变压器温度仿真_哔哩哔哩_bilibili 4、参考论文 略 大型油浸风冷变压器绕组温度场分析_高原 基于顶层油温的变压器绕组热点温度计算改进模型_陈伟根 基于热电类比理论的油浸式电…

Python及Pycharm专业版下载安装教程(Python 3.11版)附JetBrains学生认证教程

目录 一、Python下载及安装1、Python下载2、Python安装3、验证是否安装成功 二、PyCharm下载及安装1、PyCharm下载2、PyCharm安装3、激活PyCharm 三、JetBrains学生认证 本篇主要介绍Python和PyCharm专业版的下载及安装方式&#xff0c;以及通过两种方式进行JetBrains学生认证。…

基于ZYNQ的PCIE高速数据采集卡的设计(五)上位机软件设计

4.4 上位机软件设计 上位机主要完成数据的接收和保存文件等功能。由于采集卡是一种 PCIE 设备&#xff0c; 需要一种 PCIE 设备的驱动程序为基础进行开发。本设计选择 WinDriver 开发工具来 生成 PCIE 设备驱动和函数库&#xff0c;并基于该驱动在 Visual Studio(…