【Qt】信号与槽

1 🍑信号和槽概述🍑

在 Qt 中,用户和控件的每次交互过程称为⼀个事件。⽐如 “⽤⼾点击按钮” 是⼀个事件,“⽤⼾关闭窗⼝” 也是⼀个事件。每个事件都会发出⼀个信号,例如⽤⼾点击按钮会发出 “按钮被点击” 的信号,⽤⼾关闭窗⼝会发出 “窗⼝被关闭” 的信号。

Qt 中的所有控件都具有接收信号的能⼒,⼀个控件还可以接收多个不同的信号。对于接收到的每个信号,控件都会做出相应的响应动作。例如,按钮所在的窗⼝接收到 “按钮被点击” 的信号后,会做出 “关闭⾃⼰” 的响应动作;再⽐如输⼊框⾃⼰接收到 “输⼊框被点击” 的信号后,会做出 “显⽰闪烁的光标,等待⽤⼾输⼊数据” 的响应动作。在 Qt 中,对信号做出的响应动作就称之为槽

信号和槽是 Qt 特有的消息传输机制,它能将相互独⽴的控件关联起来。⽐如,“按钮” 和 "窗⼝"本⾝是两个独⽴的控件,点击 “按钮” 并不会对 “窗⼝” 造成任何影响。通过信号和槽机制,可以将 “按钮” 和 “窗⼝” 关联起来,实现 “点击按钮会使窗⼝关闭” 的效果。

信号的本质就是事件,槽的本质就是对信号响应的函数。

💡槽就是⼀个函数,与⼀般的 C++ 函数是⼀样的,可以定义在类的任何位置( public/protected/private),可以具有任何参数,可以被重载,也可以被直接调⽤(但是不能有默认参数)。槽函数与⼀般的函数不同的是:槽函数可以与⼀个信号关联,当信号被发射时,关联的槽函数被⾃动执⾏。

说明:

  • 信号和槽机制底层是通过函数间的相互调⽤实现的。每个信号都可以⽤函数来表⽰,称为信号函数;每个槽也可以⽤函数表⽰,称为槽函数。例如: “按钮被按下” 这个信号可以⽤ clicked() 函数表⽰,“窗⼝关闭” 这个槽可以⽤ close() 函数表⽰,假如使⽤信号和槽机制实现:“点击按钮会关闭窗⼝” 的功能,其实就是 clicked() 函数调⽤ close() 函数的效果。
  • 信号函数和槽函数通常位于某个类中,和普通的成员函数相⽐,它们的特别之处在于:信号函数⽤ signals 关键字修饰,槽函数⽤ public/protected/private slots 修饰。signalsslots 是 Qt 在 C++ 的基础上扩展的关键字,专⻔⽤来指明信号函数和槽函数。
  • 信号函数只需要声明,不需要定义(实现),⽽槽函数需要定义(实现)。

💡 信号函数的定义是 Qt ⾃动在编译程序之前⽣成的. 编写 Qt 应⽤程序的开发者⽆需关注,这种⾃动⽣成代码的机制称为 元编程(Meta Programming) . 这种操作在很多场景中都能⻅到。


2 🍑信号和槽的使用🍑

2.1 🍎连接信号和槽🍎

在 Qt 中,QObject 类提供了⼀个静态成员函数 connect() ,该函数专⻔⽤来关联指定的信号函数和槽函数。

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,通常不需要⼿动设定。

看一下上个博客中的代码:
在这里插入图片描述
我们从上面可以看出在connect函数中:第一个参数必须要与第二个参数匹配,就拿上面的例子第一个参数是QPushButton类型的,那么第二个参数的信号函数必须是QPushButton中的信号函数,不能够是其他类型的;第3个参数与第四个参数也是同理。

那么问题来了,我们如何去查看QPushButton中的信号函数和槽函数呢?
我们最好的方式是借助帮助文档,将光标定位到我们想要查询地方,按住F1进行跳转即可:

在这里插入图片描述
此时我们向下浏览时发现没有 click() 信号函数( 注意跟clicked() 槽函数区别),别急我们再去他的父类看看:
在这里插入图片描述
我们发现他的父类里面有,如果父类没有的话那就找父类的父类,一层一层往上找总能够找到的。

细心的同学可能还发现了一个问题,connect函数的第2个和第四个参数类型是const char *类型的,但是我们传入的是一个函数指针,这会出错的吧?

其实在早期的Qt版本中使用了一个SINGALSLOT的宏处理的,但是在Qt5后就觉得太麻烦了,便不再使用了,那么他是如何处理的呢?我们转到定义看看(快捷方式Ctrl+Enter):
在这里插入图片描述

这里面使用了Qt封装的一个类型萃取器,可以用来进行参数不匹配的检查。

2.2 🍎通过 Qt Creator 自动生成信号和槽代码🍎

Qt Creator 可以快速帮助我们⽣成信号和槽相关的代码。

我们首先双击 widget.ui ⽂件,进⼊ UI 设计界⾯,然后将按钮拖拽到绘画框里面进行按钮文字设置:

在这里插入图片描述
然后选中控件,点击鼠标右键,选择转到槽:
在这里插入图片描述
点击OK后我们返回widget.h文件中进行查看:
在这里插入图片描述
说明:
⾃动⽣成槽函数的名称有⼀定的规则。槽函数的命名规则为:on_XXX_SSS,其中:

  • on 开头,中间使⽤下划线连接起来;
  • XXX 表⽰的是对象名(控件的 objectName 属性)。
  • SSS 表⽰的是对应的信号。

如: on_pushButton_clicked() ,pushButton 代表的是对象名,clicked 是对应的信号。

widget.cpp中我们可以自己实现对应的槽函数,比如我这里实现关闭窗口操作:
在这里插入图片描述
按照这种命名⻛格定义的槽函数, 就会被 Qt ⾃动的和对应的信号进⾏连接。但是咱们⽇常写代码的时候, 除⾮是IDE ⾃动⽣成, 否则最好还是不要依赖命名规则, ⽽是显式使⽤ connect 更好。

⼀⽅⾯显式 connect 可以更清晰直观的描述信号和槽的连接关系,另⼀⽅⾯也防⽌信号或者槽的名字拼写错误导致连接失效。(当然, 是配置⼤于约定, 还是约定⼤于配置, 哪种更好, 这样的话题业界尚存在争议. 此处我个⼈还是更建议优先考虑显式 connect)

2.3 🍎自定义信号和槽🍎

在 Qt 中,允许⾃定义信号的发送⽅以及接收⽅,即可以⾃定义信号函数和槽函数。但是对于⾃定义的信号函数和槽函数有⼀定的书写规范。

⾃定义信号函数书写规范

  • ⾃定义信号函数必须写到 signals 下;
  • 返回值为 void,只需要声明,不需要实现;
  • 可以有参数,也可以发⽣重载;

⾃定义槽函数书写规范

  • 早期的 Qt 版本要求槽函数必须写到 public slots 下,但是现在⾼级版本的 Qt 允许写到类的 public 作⽤域中或者全局下;
  • 返回值为 void,需要声明,也需要实现;
  • 可以有参数,可以发⽣重载;

发送信号

使⽤ emit 关键字发送信号 。emit 是⼀个空的宏。emit 其实是可选的,没有什么含义,只是为了提醒开发⼈员。

2.3.1 🍋自定义无参数槽🍋

在这里插入图片描述
验证:
在这里插入图片描述
当按下按钮后:
在这里插入图片描述

2.3.2 🍋自定义无参数信号🍋

我们使用可视化方式生成一个按钮,然后转到槽,构建一个自定义的信号并且在on_pushButton_clicked()函数中发送mySignal()信号:
在这里插入图片描述
然后运行:
在这里插入图片描述
点击按钮:
在这里插入图片描述
发现验证成功。

在实际业务中,我们很少会自定义信号,因为Qt默认给我们提供的信号已经能够处理绝大部分的场景了,而槽函数的话一般情况下都是会要用户自定义生成的。

2.3.3 🍋自定义有参数槽和信号🍋

Qt 的信号和槽也⽀持带有参数, 同时也可以⽀持重载。此处我们要求, 信号函数的参数列表要和对应连接的槽函数参数列表⼀致,此时信号触发, 调⽤到槽函数的时候, 信号函数中的实参就能够被传递到槽函数的形参当中。

💡 通过这样的机制, 就可以让信号给槽传递数据了。

但是通过自定义有参数槽和信号时要注意:信号与槽的类型要一致并且信号的参数>=槽的参数个数

我们来举一个简单的例子:
在这里插入图片描述
在这里插入图片描述
当我们运行时:
在这里插入图片描述
我们每点击一次就会运行一次,当我们增加自定义信号的参数时:
在这里插入图片描述
发现也不会报错。

2.4 🍎信号与槽的连接方式🍎

2.4.1 🍋⼀对⼀🍋

主要有两种形式,分别是:⼀个信号连接⼀个槽⼀个信号连接⼀个信号

一个信号关联一个槽很好理解,前面我们写的代码几乎都是这样,而一个信号关联一个信号我们可以来验证下:
在这里插入图片描述

通过这种方式让一个按钮的信号关联:handButton(),然后在handButton()发射mySignal()信号,最后调用myHander()槽函数来进行实际的处理。

在这里插入图片描述

2.4.2 🍋⼀对多🍋

⼀个信号连接多个槽。

2.4.3 🍋多对⼀🍋

多个信号连接⼀个槽函数。

2.4.4 🍋多对多🍋

多个信号连接多个槽函数。

2.5 🍎信号与槽的断开🍎

使⽤ disconnect 即可完成断开。disconnect 的⽤法和 connect 基本⼀致。

2.6 🍎使⽤ Lambda 表达式定义槽函数🍎

Qt5 在 Qt4 的基础上提⾼了信号与槽的灵活性,允许使⽤任意函数作为槽函数。
但如果想⽅便的编写槽函数,⽐如在编写函数时连函数名都不想定义,则可以通过 Lambda表达式 来达到这个⽬的。

Lambda表达式 是 C++11 增加的特性。C++11 中的 Lambda表达式 ⽤于定义并创建匿名的函数对象,以简化编程⼯作。(不清楚语法格式的可以看看博主之前讲解过的文章)


3 🍑信号与槽的优缺点🍑

优点: 松散耦合

信号发送者不需要知道发出的信号被哪个对象的槽函数接收,槽函数也不需要知道哪些信号关联了⾃⼰,Qt的信号槽机制保证了信号与槽函数的调⽤。⽀持信号槽机制的类或者⽗类必须继承于 QObject类。

缺点: 效率较低

与回调函数相⽐,信号和槽稍微慢⼀些,因为它们提供了更⾼的灵活性,尽管在实际应⽤程序中差别不⼤。通过信号调⽤的槽函数⽐直接调⽤的速度慢约10倍(这是定位信号的接收对象所需的开销;遍历所有关联;编组/解组传递的参数;多线程时,信号可能需要排队),这种调⽤速度对性能要求不是⾮常⾼的场景是可以忽略的,是可以满⾜绝⼤部分场景。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

LabVIEW 2024安装教程(附免费安装包资源)

鼠标右击软件压缩包,选择“解压到LabVIEW.2024”。 返回解压后的文件夹,鼠标右击“ni_labview-2024”选择“装载”。 鼠标右击“Install”选择“以管理员身份运行”。 点击“我接受上述2条许可协议”,然后点击“下一步”。 点击“下一步”。 …

用例整体执行及pytest.ini文件

在我们写代码的过程中,一般都是右键或者命令行去执行一个用例 但是当我们写完后,需要整体执行一遍。那应该怎么搞呢? 我们可以在根目录下新建一个main.py或者run.py之类的文件,文件内容如下: if __name__ "__ma…

Stable Diffusion 常用放大算法详解

常用放大算法 图像放大算法大致有两种: 传统图像放大算法(Lantent、Lanczos、Nearest)AI图像放大算法(4x-UltraSharp、BSRGAN、ESRGAN等) 传统图像放大算法是基于插值算法,计算出图像放大后新位置的像素…

Redis 源码学习记录:字符串

redisObject Redis 中的数据对象 server/redisObject.h 是 Redis 对内部存储的数据定义的抽象类型其定义如下: typedef struct redisObject {unsigned type:4; // 数据类型,字符串,哈希表,列表等等unsigned encoding:4; …

微信小程序:9.小程序配置

全局配置文件 小程序根目录下的app.json文件是小程序的全局配置文件。 常用的配置文件如下: pages 记录当前小程序所有的页面存放路径信息 window 全局设置小程序窗口外观 tabBar 设置小程序底部的tabBar效果 style 是否启用新版style 小程序窗口的组成部分 了解windo节点常…

keytool,openssl的使用

写在前面 在生成公钥私钥,配置https时经常需要用到keytool,openssl工具,本文就一起看下其是如何使用的。 keytool是jdk自带的工具,不需要额外下载,但openssl需要额外下载 。 1:使用keytool生成jks私钥文件…

ArcGIS专题图制作—3D峡谷地形

6分钟教你在ArcGIS Pro中优雅完成炫酷的美国大峡谷3D地图 6分钟教你在ArcGIS Pro中优雅完成炫酷的美国大峡谷3D地图。 这一期的制图教程将带我们走入美国大峡谷,让我们一起绘制这张美妙的地图吧!视频也上传到了B站,小伙伴可以去! …

每日一题(力扣45):跳跃游戏2--贪心

由于题目已经告诉了我们一定可以跳到,所以我们只需去考虑前进最快的方法。即 判断当前下一步能跳的各个位置中,哪个能带你去去向最远的地方(why? 因为其他位置所能提供的最大范围都没最远那个大,所以最远的那个已经可以…

IBM SPSS Statistics for Mac v27.0.1中文激活版:强大的数据分析工具

IBM SPSS Statistics for Mac是一款功能强大的数据分析工具,为Mac用户提供了高效、精准的数据分析体验。 IBM SPSS Statistics for Mac v27.0.1中文激活版下载 该软件拥有丰富的统计分析功能,无论是描述性统计、推论性统计,还是高级的多元统计…

上门服务系统|上门服务小程序搭建流程

随着科技的不断进步和人们生活水平的提高,越来越多的服务开始向线上转型。传统的上门服务业也不例外,随着上门服务小程序的兴起,人们的生活变得更加便捷和高效。本文将为大家介绍上门服务小程序的搭建流程以及应用范围。 一、上门服务小程序搭…

ZYNQ之嵌入式开发04——自定义IP核实现呼吸灯、固化程序

文章目录 自定义IP核——呼吸灯实验固化程序 自定义IP核——呼吸灯实验 Xilinx官方提供了很多IP核,在Vivado的IP Catalog中可以查看这些IP核,在构建自己复杂的系统时,只使用Xilinx官方的免费IP核一般满足不了设计的要求,因此很多…

浅谈游戏机制

浅谈游戏机制 前言什么是游戏机制?机制组成机制类别结语 前言 最近在编写游戏开发文档的时候了解到游戏机制,第一次接触游戏机制的概念难免有些陌生,但感觉又跟常见,在网上查阅浏览了一些资料后了解到游戏机制还不止一个。 现在将…

微信小程序:12.页面导航

什么是页面导航 页面导航指的是页面之间的相互跳转。例如,浏览器中实现的页面导航的方式有两种: 连接location.href 小程序中实现页面导航的两种方式 声明式导航 在页面上声明一个导航组件 通过点击组件实现页面跳转 导航TabBar页面 是指配置TabB…

HarmonyOS ArkUI实战开发—状态管理

一、状态管理 在声明式UI编程框架中,UI是程序状态的运行结果,用户构建了一个UI模型,其中应用的运行时的状态是参数。当参数改变时,UI作为返回结果,也将进行对应的改变。这些运行时的状态变化所带来的UI的重新渲染&…

店匠科技技术产品闪耀,引领新质生产力发展

在科技飞速发展的今天,新质生产力正成为推动社会进步和经济高质量发展的核心力量。店匠科技,作为一家致力于为全球B2C电商提供产品和技术解决方案的领先企业,其技术产品不仅体现了新质生产力的创新特质,更在推动电商行业转型升级中发挥了重要作用。 新质生产力,以创新为主导,摆…

嵌入式开发一:初识Stm32

目录 一、嵌入式简介 1.1 嵌入式概念 1.2 嵌入式系统的组成 1.3 嵌入式的分类 1.3.1 嵌入式系统的分类 1.3.2 嵌入式处理器的分类 二、单片机简介(MCU嵌入式微控制器) 2.1 单片机是什么 2.2 单片机的作用是什么 2.3 单片机的发展历程 2.4 单片机发展趋势 2.5 复杂指…

将图片添加描述批量写入excel

原始图片 写入excel的效果 代码 # by zengxy chatgpt # from https://blog.csdn.net/imwatersimport os import xlsxwriter from PIL import Imageclass Image2Xlsx():def __init__(self,xls_path,head_list[编号, 图片, 名称, "描述",备注],set_default_y112,se…

Java毕业设计 基于SpringBoot vue城镇保障性住房管理系统

Java毕业设计 基于SpringBoot vue城镇保障性住房管理系统 SpringBoot 城镇保障性住房管理系统 功能介绍 首页 图片轮播 房源信息 房源详情 申请房源 公示信息 公示详情 登录注册 个人中心 留言反馈 后台管理 登录 个人中心 修改密码 个人信息 用户管理 房屋类型 房源信息管理…

Eudic欧路词典for Mac:专业英语学习工具

Eudic欧路词典for Mac,作为专为Mac用户设计的英语学习工具,凭借其简捷高效的特点,成为众多英语学习者不可或缺的助手。 Eudic欧路词典for Mac v4.6.4激活版下载 这款词典整合了多个权威词典资源,如牛津、柯林斯、朗文等&#xff0…

2024 java easyexcel poi word模板填充数据,多个word合成一个word

先看效果 一、准备工作 1.word模版 2.文件路径 二、pom依赖 <!-- easyexcel --><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>2.1.7</version></dependency><depe…