[Qt] 信号和槽(1) | 本质 | 使用 | 自定义

目录

一、信号和槽概述

二、本质

底层实现

1. 函数间的相互调用

2. 类成员中的特殊角色

三、使用

四. 自定义信号和槽

1. 基本语法

(1) 自定义信号函数书写规范

(2) 自定义槽函数书写规范

(3) 发送信号

(4) 示例

A. 示例一

B. 示例二 —— 老师说“上课了”,学生回到座位学习

C. 示例三 —— 老师点击按钮触发学生上课


一、信号和槽概述

  • 事件与信号
    • 在 Qt 中,用户与控件的交互(如点击按钮或关闭窗口)产生事件,每个事件触发相应的信号。
    • 信号是事件的通知形式,通过函数表示。
  • 响应与槽
    • 控件接收信号并作出响应动作,称为槽。
    • 槽为普通 C++ 函数,可定义在类的不同访问级别中,并能被关联到一个或多个信号上自动执行。


二、本质

1. 信号的本质

信号本质上是事件,由用户操作触发,通过特定函数形式通知开发者。

例如:

  • 按钮单击、双击
  • 鼠标移动、鼠标按下、鼠标释放
  • 键盘输入

那么在 Qt 中信号是通过什么形式呈现给使用者的呢?

  • 我们对哪个窗口进行操作, 哪个窗口就可以捕捉到这些被触发的事件。
  • 对于使用者来说触发了一个事件,我们就可以得到 Qt 框架给我们发出的某个特定信号。
  • 信号的呈现形式就是函数, 也就是说某个事件产生了, Qt 框架就会调用某个对应的信号函数,通知使用者。

2. 槽的本质

槽是回调函数,用于响应信号,当关联的信号发射时,槽函数会被自动调用。

关于 回调函数

C 语言阶段 - 函数指针

  • 实现转移表,降低代码的 “圈复杂度”
  • 实现回调函数效果(qsort)

C++ 阶段

  • STL 中,函数对象 / 仿函数
  • lambda 表达式

Linux

  • 信号处理函数
  • 线程的入口函数
  • epoll 基于回调的机制

底层实现

1. 函数间的相互调用

在Qt框架中,信号和槽机制本质上是通过函数间的相互调用来实现的。具体来说:

  • 信号函数:每个信号都可以用一个函数来表示。例如,“按钮被按下”的事件可以用clicked()信号函数来表示。
  • 槽函数:同样地,每个槽也可以用一个函数表示,比如“窗口关闭”的动作可以用close()槽函数来表示。

当使用信号和槽机制来实现特定功能时,如点击按钮关闭窗口,实际上是在连接clicked()信号与close()槽函数,使得点击按钮(触发clicked())会导致窗口关闭(执行close())的效果。

2. 类成员中的特殊角色

信号函数和槽函数通常位于某个类中,但它们与普通成员函数相比有一些特别之处:

  • 修饰关键字

这些关键字是Qt对C++语言的扩展,专门用于标识信号和槽函数

    • 信号函数使用signals关键字进行修饰。
    • 槽函数则使用public slotsprotected slotsprivate slots关键字修饰。
  • 声明与定义
    • 信号函数只需要声明而无需定义;其定义由Qt自动处理,这种自动生成代码的机制称为 “ 元编程(Meta Programming),这种操作在很多场景中都能见到。
    • 槽函数需要像普通函数一样定义和实现,因为它们包含了具体的响应逻辑。

三、使用

1. 连接信号和槽

  • QObject 是 Qt 内置的父类,Qt 中提供的很多类都是直接或者间接继承自 QObject。(在 Java 中也存在相似的设定)
  • 使用 QObject::connect() 函数关联信号与槽,指定发送者、信号类型、接收者及方法。
connect (const QObject *sender,    //描述当前信号是哪个控件发出的const char * signal ,      //信号类型const QObject * receiver , //信号如何处理(哪个对象处理)const char * method ,      //信号如何处理(这个对象该怎么处理——要处理信号的对象提供的成员函数)Qt::ConnectionType type = Qt::AutoConnection ) //暂时不考虑,很少使用

参数说明

  • sender:信号的发送者。
  • signal:发送的信号(信号函数)。
  • receiver:信号的接收者。
  • method:接收信号的槽函数
  • type:用于指定关联方式,默认的关联方式为 Qt::AutoConnection,通常不需要手动设定。

示例:

在窗口中设置一个按钮,当点击 “按钮” 时关闭 “窗口”:

tip:

2. 查看内置信号和槽

  • 利用 Qt 帮助文档查询系统自带的信号和槽,通常在类或其父类中查找。

如上述示例,要查询 “按钮” 的信号,在帮助文档中输入:QPushButton

首先可以在 "Contents" 中寻找关键字 signals如果没有找到,继续去父类中查找,因此我们去他的父类 QAbstractButton 中继续查找关键字 signals

这里的 clicked() 就是我们要找的信号。槽函数的寻找方式和信号一样,只不过它的关键字是 slot。


3. 通过 Qt Creator 生成代码

  • Qt Creator 提供可视化界面帮助快速生成信号槽代码,简化开发流程。

示例:点击按钮关闭窗口

  • 新建项目并创建 UI 文件。
  • 在 UI 设计界面添加按钮并配置属性。

可视化生成槽函数

当鼠标单击 “转到槽...” 之后,出现如下界面:对于按钮来说,当点击时发送的信号是:clicked(),所以此处选择:clicked()

这个窗口列出了 QPushButton 给我们提供的所有信号(还包含了 QPushButton 父类的信号)。

对于普通按钮来说,使用 clicked 信号即可。

clicked(bool) 没有意义的,具有特殊状态的按钮(比如:复选按钮)才会用到 clicked(bool)。

  • 然后 就会跳转到 Qt Creator 自动生成槽函数原型框架,并在槽函数中添加关闭窗口逻辑。

说明:自动生成槽函数的名称有一定的规则,槽函数的命名规则为:on_XXX_SSS,其中:

  1. 以 "on" 开头,中间使用下划线连接起来。
  2. "XXX" 表示的是对象名(控件的 objectName 属性)。
  3. "SSS" 表示的是对应的信号

如:"on_pushButton_clicked() ",pushButton 代表的是对象名,clicked 是对应的信号。按照这种命名风格定义的槽函数,就会被 Qt 自动的和对应的信号进行连接

但是我们日常写代码的时候,除非是 IDE 自动生成,否则最好还是不要去依赖命名规则,而是显式使用 connect 更好。

  • 一方面,显式 connect 可以更清晰直观的描述信号和槽的连接关系。
  • 另一方面,也是防止信号或者槽的名字拼写错误导致连接失效。

当然,是配置大于约定,还是约定大于配置,哪种更好。这样的话题业界尚存在争议,这里我个人还是更建议 日常优先考虑显式 connect。

"widget.cpp" 中自动生成槽函数定义 ,在槽函数函数定义中添加要实现的功能,实现关闭窗口的效果


四. 自定义信号和槽

1. 基本语法

在 Qt 中,允许开发者自定义信号的发送方及接收方,即 可以定义自己的信号函数和槽函数。为了实现这一点,必须遵循一定的 书写规范。

  • Q_OBJECT 宏
    • 在类中使用信号槽机制的前提是类最开始处需要声明Q_OBJECT宏。这个宏展开大量代码,使编译器能够识别并处理信号和槽。
    • Qt Creator通常会自动添加此宏。

(1) 自定义信号函数书写规范

规范要点

  • 自定义信号较少见,因为Qt内置信号已足够满足大多数需求。
  • 信号本质上是特殊的函数,仅需声明无需实现,返回类型为void
  • 可以有参数,并支持重载。
  • 必须放在signals关键字下,这是Qt扩展的关键字,用于标识信号。

构建过程

头文件的 关键字下 定义

  • qmake工具在构建项目时会扫描到signals关键字,并自动生成相应的函数定义。
(2) 自定义槽函数书写规范
  • 操作规程
    1. 槽函数的定义与普通成员函数无异,但早期版本要求槽函数必须位于public slotsprotected slotsprivate slots下。
    2. 现代版本允许槽函数放置于public作用域中或全局范围内。
    3. 返回类型为void,需要声明和实现,支持参数和重载。
  • slots 关键字

    • 同样是Qt扩展的关键字,用于指示qmake生成相关代码。
    • 我们上面有提到过,会进行 元编程
(3) 发送信号
  • 使用emit关键字来触发信号。
  • 尽管emit是一个空宏且可选,但它有助于提醒开发人员当前正在发射信号。
(4) 示例
A. 示例一
  • widget.h中声明自定义信号和槽。

  • widget.cpp中实现槽函数,并连接信号和槽。

运行

B. 示例二 —— 老师说“上课了”,学生回到座位学习
  • 创建老师类和学生类,分别声明信号和槽。

在源文件中新建两个类,一个是老师类,一个是学生类;首先选中项目名称,鼠标右键 ——> "add new..."

注意:在 Qt 中新建类时,要选择新建类的父类。

  • 显然,当前项目中还没有什么类适合做新类的父类,同时新的类也不是一个 “窗口” 或者 “控件”,这种情况一般选择 QObject 作为基类。
  • 这样做的好处是这个新类的对象可以搭配 Qt 的对象树机制,便于对象的正确释放。

对于 “学生类” 以上述同样的方式进行添加,添加完成之后,项目目录新增文件如下:

  • widget.h中实例化这两个类的对象。

  • student.cpp中实现槽函数。

  • widget.cpp中连接信号和槽。

运行

C. 示例三 —— 老师点击按钮触发学生上课
  • 实现逻辑同上,通过UI中的按钮触发事件。

运行:

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

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

相关文章

OpenGL变换矩阵和输入控制

在前面的文章当中我们已经成功播放了动画,让我们的角色动了起来,这一切变得比较有意思了起来。不过我们发现,角色虽然说是动了起来,不过只是在不停地原地踏步而已,而且我们也没有办法通过键盘来控制这个角色来进行移动…

overscroll-behavior-解决H5在ios上过度滚动的默认行为

1. 问题 开发H5的过程中,经常会有android和ios两边系统需要兼容的情况。在ios上一直有个问题是当H5内容触及到页面顶部或底部时,还是可以被人为的往下或往下拉动界面。当然可能有的情况是比较适用的,比如你往下拉动,然后在导航栏…

复杂对象的创建与组装 - 建造者模式(Builder Pattern)

建造者模式(Builder Pattern) 建造者模式(Builder Pattern)建造者模式(Builder Pattern)概述建造者模式结构图代码 talk is cheap, show you my code总结 建造者模式(Builder Patter…

Linux-mac地址

mac地址 由6位16进制数组成。最高字节的最低位,0表示单播地址,1表示多播地址。最高字节的第二位,0表示全局地址,1表示本地地址。 单播地址:单播MAC地址用于一对一的通信模式,即从单一的源端发送到单一的目…

SAP学习笔记 - 豆知识14 - Msg 番号 M7562 - 取引Type WL 对应的番号範囲中不存在2025年度 OMBT

这种类似的以前也写过,原因就是自动採番的番号没弄。 比如跨年了,那该新年度的番号范围没弄啊,就会出这种错误。 把番号范围给加一下就可以了。 1,现象 比如点 VL02N 出荷传票变更 画面,点 出库确认 就会出如下错误…

一文理清JS中获取盒子宽高各方法的差异

前言 这段时间在研究一个反爬产品,环境检测用到了很多个盒子宽高取值方法,如window.outerWidth、window.screen.availWidth,各个方法取值结果不大相同,在此记录下遇到的方法。 各宽方法区别 这里就讲解下各宽度方法的区别&…

sqoop将MySQL数据导入hive

使用脚本加载数据 MySQL有一张表 hive创建一张相同的表 编写脚本同步数据 [rootmaster sqoop]# vim stu.sh#!/bin/bash SQOOP/usr/local/soft/sqoop-1.4.6/bin/sqoop $SQOOP import --connect jdbc:mysql://192.168.67.100:3306/sqoop \--username root \--password 123456 \-…

Docker Compose编排

什么是 Docker Compose? Docker Compose 是 Docker 官方推出的开源项目,用于快速编排和管理多个 Docker 容器的应用程序。它允许用户通过一个 YAML 格式的配置文件 docker-compose.yml 来定义和运行多个相关联的应用容器,从而实现对容器的统一管理和编…

[羊城杯 2024]hiden

一顿解压之后发现有两个文件: 尝试了Rot47解密,得到一个看起来挺像一回事的解码结果: 再将得到的解码结果试试Rot13解密,成功得到正确的解码结果: import wave with open(flag.txt, rb) as f:txt_data f.read()file_l…

LeetCode - 初级算法 数组(只出现一次的数字)

只出现一次的数字 这篇文章讨论如何找到一个数组中只出现一次的数字,确保算法的时间复杂度为线性,且只使用常量额外空间。 免责声明:本文来源于个人知识与公开资料,仅用于学术交流。 描述 给定一个非空整数数组 nums,除了某个元素只出现一次以外,其余每个元素均出现两…

【谷歌开发者月刊】十二月精彩资讯回顾,探索科技新可能

我们在今年的尾声中回顾本月精彩,开发者们借助创新技术为用户打造温暖的应用体验,展现技术与实用的结合。欢迎您查阅本期月刊,掌握最新动态。 本月看点 精彩看点多多,请上下滑动阅览 01DevFest 北京站和上海站圆满举办&#xff0c…

LinuxC高级day4

作业: 1.思维导图 2.终端输入一个C源文件名(.c结尾)判断文件是否有内容,如果没有内容删除文件,如果有内容编译并执行改文件。 3.终端输入两个文件名,判断哪个文件的时间戳更新

数据中台与数据治理服务方案[50页PPT]

本文概述了数据中台与数据治理服务方案的核心要点。数据中台作为政务服务数据化的核心,通过整合各部门业务系统数据,进行建模与加工,以新数据驱动政府管理效率提升与政务服务能力增强。数据治理则聚焦于解决整体架构问题,确保数据…

MAC环境安装(卸载)软件

MAC环境安装(卸载)软件 jdknode安装node,并实现不同版本的切换背景 卸载node从node官网下载pkg安装的node卸载用 homebrew 安装的node如果你感觉删的不够干净,可以再细分删除验证删除结果 jdk 1.下载jdk 先去官网下载自己需要的版…

时间序列预测算法---LSTM

文章目录 一、前言1.1、深度学习时间序列一般是几维数据?每个维度的名字是什么?通常代表什么含义?1.2、为什么机器学习/深度学习算法无法处理时间序列数据?1.3、RNN(循环神经网络)处理时间序列数据的思路?1.4、RNN存在哪些问题?…

LinuxC高级day2

1.在家目录下创建目录文件,dir a.dir下创建dir1和dir2 b.把当前目录下的所有文件拷贝到dir1中, c.把当前目录下的所有脚本文件拷贝到dir2中 d.把dir2打包并压缩为dir2.tar.xz e.再把dir2.tar.xz移动到dir1中 f.解压dir1中的压缩包 g.使用tree工具&#x…

14. 日常算法

1. 面试题 02.04. 分割链表 题目来源 给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。 你不需要 保留 每个分区中各节点的初始相对位置。 class Solution { public:ListNo…

ubuntu 如何使用vrf

在Ubuntu或其他Linux系统中,您使用ip命令和sysctl命令配置的网络和内核参数通常是临时的,这意味着在系统重启后这些配置会丢失。为了将这些配置持久化,您需要采取一些额外的步骤。 对于ip命令配置的网络接口和路由,您可以将这些配…

SpringMVC进阶(自定义拦截器以及异常处理)

文章目录 1.自定义拦截器 1.基本介绍 1.说明2.自定义拦截器的三个方法3.流程图 2.快速入门 1.Myinterceptor01.java2.FurnHandler.java3.springDispatcherServlet-servlet.xml配置拦截器4.单元测试 3.拦截特定路径 1.拦截指定路径2.通配符配置路径 4.细节说明5.多个拦截器 1.执…

Mac电脑python多版本环境安装与切换

我当前是python3.9.6环境,需要使用3.9.8环境,通过brew安装3.9.8版本,然后通过pyenv切换环境 步骤 1: 安装 pyenv brew install pyenv brew install pyenv-virtualenv 步骤 2: 安装 Python 3.9.8(使用 pyenv 安装指定版本的 Pyth…