【Qt学习笔记】Qt Creator环境下 信号与槽 详解(自定义信号槽、断连、lambda表达式等)

文章目录

  • 1. 信号槽概念
    • 1.1 信号的本质
    • 1.2 槽的本质
    • 1.3 标准信号槽
    • 1.4 信号槽 实例
  • 2. 自定义信号槽
    • 2.1 自定义槽函数
    • 2.2 自定义信号
    • 2.3 带参 信号槽
  • 3. 信号槽的意义 与 作用
  • 4. 信号槽断连 (了解)
  • 5. lamda表达式的使用
    • 5.1 基本用法
    • 5.2 捕获局部变量
    • 5.3 使用捕获列表捕获变量

1. 信号槽概念

信号槽 是 Qt 框架中一种用于对象间通信的机制。它通过让一个对象发出信号,另一个对象连接到这个信号的槽上来实现通信。信号槽机制是 Qt 的核心特性之一,提供了一种灵活且类型安全的方式来处理事件和数据传递。


1.1 信号的本质

QT中,信号由三部分组成:

  1. 信号源:哪个控件发出的信号
  2. 信类别:什么类别的信号
  3. 信号处理方式:通过信号处理函数,当信号被触发时执行。

1.2 槽的本质

本身就是一个成员函数,负责对QT中产生的信号进行处理。

在编写代码时,槽函数的编写等同于类的成员函数。

举例

  • 有一个按钮,当点击这个按钮时,会关闭当前窗口。
  • connect(pushButton, &QPushButton::clicked, this, &MainWindow::close)
    对于该例来说:
    • 信号源:按钮 pushButton
    • 信号类别:点击信号 clicked
    • 信号处理方式:利用连接connect 与槽函数slot(即close)进行处理

1.3 标准信号槽

即 Qt框架 中内置的信号槽

Qt框架中,存在一些标准信号(Standard Signals)和标准槽(Standard Slots)的已定义信号和槽函数。这些标准信号和槽提供了一些常见的功能,可以在不同的Qt对象之间进行连接。

  • 标准信号如:QPushButton、QLineEdit、QSlider等。
  • 标准槽如:对于QWidget,有show()、hide()、close()等。

1.4 信号槽 实例

我们尝试写一个关闭窗口按钮

举例

我们将使用的标准信号 以及 槽函数:

// 单击按钮后发出的信号
[signal] void QAbstractButton::clicked(bool checked = false)
// 关闭窗口的槽函数
[slot] bool QWidget::close();

我们需要在mainWindow.ui中插入一个QPushButton类(不要忘记更改objectName)
在这里插入图片描述

随后通过 connect函数 将该功能实现出来:

// 单击按钮关闭窗口
connect(ui->closeBtn, &QPushButton::clicked, this, &MainWindow::close);

在上面的例子中:

  • 信号(Signal):&QPushButton::clicked。这里的信号是QPushButton类的clicked信号。当用户点击按钮时,QPushButton对象会发出(emit)一个clicked信号。

  • 槽(Slot):&MainWindow::close。这里的槽是MainWindow类的close成员函数。槽可以是任何普通的成员函数,它被设计为响应特定信号的函数。


2. 自定义信号槽

当QT提供的标准信号槽无法满足需求时,我们可以设计所需的信号与槽的功能,最后通过connect连接以实现功能。

当进行自定义信号槽时,需要遵循以下规则:

  • 当我们编写新的类时,要让其继承Qt的某些标准类
    • 继承自QObject:类必须直接或间接地继承自QObject,以便能够使用信号和槽机制
    • Q_OBJECT 宏:在类的声明中,使用Q_OBJECT宏进行标记。
    • 信号与槽函数声明:在类内部声明自定义信号和槽函数。信号声明只需要在signals部分声明,槽函数声明则需要在public slots或private slots部分声明
// 在头文件派生类的时候,首先如下面的写法引入Q_OBJECT宏:
class MyMainWindow : public QWidget
{Q_OBJECT// ... ...
}

2.1 自定义槽函数

自定义槽函数有两种写法:

法一

一种是当我们使用代码创建控件时,自定义槽函数后手动连接以及实现。

在这里插入图片描述
随后我们在widget.cpp中手动进行创建按钮以及连接信号的操作。

在这里插入图片描述
结果显示:

在这里插入图片描述


法二

我们通过图形化界面创建控件,并利用Qt Creator的功能自动连接。

在这里插入图片描述
此时Qt Creator会自动生成槽函数的定义与声明:

在这里插入图片描述
且此时,我们只需要实现 on_pushButton_clicked() 函数,不需要进行connect就可以实现按钮的功能。

原因如下:

在这里插入图片描述


2.2 自定义信号

以下是一些自定义信号时的要求注意事项:

  1. 返回值必须是 void 类型(直接不写返回值就行)
  2. 信号需要使用 signals 关键字进行声明, 使用方法类似于public等
  3. 信号函数只需要声明, 不需要定义(没有函数体实现)
  4. 如要要发射自定义信号,一般在信号函数前加emit ,表示发射
    • 发送信号的本质就是调用信号函数
    • emit 关键字只是发出信号的标志,没有特殊含义,可以省略

举例

我们首先在头文件中定义信号
在这里插入图片描述
再用connect将自定义信号与槽连接起来、可以直接在构造函数中发送信号,也可以通过其他函数发送信号,我们这里通过按钮按下的槽函数发送信号。
在这里插入图片描述


2.3 带参 信号槽

信号和槽都可以带有参数、自然也有函数重载,两者之间满足规则:

信号和槽的参数类型需一致,且信号的参数个数不能少于槽(可以多于)。

在这里插入图片描述


3. 信号槽的意义 与 作用

  1. 解耦合
    在传统的编程模式中,一个对象A直接调用另一个对象B的函数,形成强耦合关系。如果对象B的函数接口发生变化,那么对象A也需要相应地修改。这种直接依赖关系使得代码难以维护和扩展。信号槽机制通过信号和槽的连接,实现了对象间的间接通信,降低了对象之间的耦合度。

  2. 易于扩展和维护
    由于信号和槽减少了对象间的直接依赖,当系统需要新增功能或者修改现有功能时,往往只需要添加或修改相关的信号和槽,而不需要对其他组件进行大幅度修改。这大大提高了代码的可维护性和可扩展性。

  3. 支持异步通信
    在传统的同步调用中,调用方必须等待被调用方处理完成后才能继续执行,这在某些情况下会导致程序的响应性不佳。信号槽机制支持异步通信,即发送信号后,发送方可以继续其它操作,而不需要等待接收方处理完毕。这对于提高程序的响应性和性能尤为重要。


4. 信号槽断连 (了解)

Qt中可以使用connect连接信号与槽、同时也可以使用disconnect 断开某个信号与槽的连接

disconnect()函数有几种重载形式,可以根据需要选择使用。

由于一般不用disconnect 断开连接,下面进行简单举例:

示例1:断开特定信号和槽
假设有一个按钮(QPushButton)和一个标签(QLabel),当按钮被点击时,标签的文本会改变。如果在某个时刻想要停止这种行为,可以使用disconnect()断开它们之间的连接:

QPushButton *button = new QPushButton("Click me");
QLabel *label = new QLabel("Hello");// 连接信号和槽
QObject::connect(button, &QPushButton::clicked, [label]() {label->setText("Button Clicked!");
});// 假设在某个条件下,我们需要断开上面建立的连接
QObject::disconnect(button, &QPushButton::clicked, nullptr, nullptr);

这里,disconnect()的调用断开了button的clicked信号与所有槽之间的连接。

示例2:断开所有与对象相关的连接
如果你想断开一个对象的所有信号与槽的连接,可以简单地传递该对象作为参数给disconnect():

// 断开与button相关的所有信号和槽的连接
QObject::disconnect(button);

这将断开button发出的所有信号与任何槽之间的连接,同时也断开任何信号到button槽的连接。

示例3:断开特定的信号和特定的槽
如果你只想断开一个特定信号与一个特定槽之间的连接,可以这样做:

// 假设有一个自定义槽函数
void customSlot();// 连接信号和槽
QObject::connect(button, &QPushButton::clicked, this, &MyClass::customSlot);// 在某个条件下,只断开这个特定的信号和槽的连接
QObject::disconnect(button, &QPushButton::clicked, this, &MyClass::customSlot);

这里,只有button的clicked信号与MyClass的customSlot槽之间的连接被断开。

示例4:使用返回值断开连接
connect()函数返回一个QMetaObject::Connection对象,可以用来在稍后断开连接:

QMetaObject::Connection conn = QObject::connect(button, &QPushButton::clicked, []() {qDebug() << "Button clicked!";
});// 断开连接
QObject::disconnect(conn);

这种方式允许对特定的连接进行更精确的控制。

使用disconnect()可以灵活地管理信号和槽之间的连接状态,根据应用程序的需要动态调整其行为。


5. lamda表达式的使用

这里是关于lamda表达式的一篇文章:

【C++11】lambda表达式 的定义、性质和用法

在Qt 5及以上版本中,connect()函数支持使用lambda表达式作为槽函数。这使得在连接信号和槽时可以直接在参数中编写逻辑处理代码,而不需要定义额外的槽函数。这种方式可以让代码更加紧凑和灵活,特别是当槽函数只在一个地方使用且逻辑简单时。

5.1 基本用法

下面是一个使用lambda表达式作为槽函数的基本示例:

QPushButton *button = new QPushButton("Click me");
QObject::connect(button, &QPushButton::clicked, []() {qDebug() << "Button was clicked!";
});

在该例中,当按钮被点击时,会执行lambda表达式内的代码,即打印一条消息到调试控制台。

5.2 捕获局部变量

Lambda表达式可以捕获上下文中的变量,以便在表达式内部使用。例如,假设我们想在按钮点击时改变一个标签(QLabel)的文本:

QPushButton *button = new QPushButton("Change Label");
QLabel *label = new QLabel("Original Text");QObject::connect(button, &QPushButton::clicked, [label]() {label->setText("Text after click");
});

这里,lambda表达式通过捕获label指针,可以在按钮被点击时修改标签的文本。

5.3 使用捕获列表捕获变量

我们知道:Lambda表达式的捕获列表允许以不同的方式捕获变量,例如按值捕获(拷贝)或按引用捕获。使用=捕获所有局部变量的副本,使用&按引用捕获所有局部变量。

int count = 0;
QPushButton *button = new QPushButton("Increase Count");QObject::connect(button, &QPushButton::clicked, [=]() mutable {qDebug() << "Count:" << ++count;
});
  • 注意:由于默认情况下lambda表达式是不允许修改捕获的变量的,如果想在lambda表达式内修改按值捕获的变量,需要在参数列表后添加mutable关键字。但是,这里的count变量实际上是按值捕获的副本,外部的count变量不会被修改。为了确保外部变量也被修改,应该按引用捕获:
int count = 0;
QPushButton *button = new QPushButton("Increase Count");QObject::connect(button, &QPushButton::clicked, [&]() {qDebug() << "Count:" << ++count;
});
  • 使用lambda表达式作为槽函数可以增加代码的灵活性和简洁性,尤其适用于处理简单的逻辑或者当我们不想为了一个简单的操作而专门去定义一个槽函数的情况。

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

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

相关文章

JavaScript相关(一)——作用域

本篇将从JS的执行上下文开始&#xff0c;去理解&#xff1a;变量提升、 栈式调用、作用域和闭包。 参考&#xff1a; 浏览器工作原理与实践 JS执行上下文 执行上下文是 JavaScript 执行一段代码时的运行环境&#xff0c;比如调用一个函数&#xff0c;就会生成这个函数的执行…

『运维备忘录』之 Ansible 自动化运维工具

一、简介 Ansible是基于Python开发&#xff0c;集合了众多运维工具&#xff08;puppet、cfengine、chef、func、fabric&#xff09;的优点&#xff0c;实现了批量系统配置、批量程序部署、批量运行命令等功能的自动化运维工具&#xff0c;广泛用于配置管理、应用部署以及任务协…

ES节点故障的容错方案

ES节点故障的容错方案 1. es启动加载逻辑1.1 segment和translg组成和分析1.2 es节点启动流程1.3 es集群的初始化和启动过程 2. master高可用2.1 选主逻辑2.1.1 过滤选主的节点列表2.1.2 Bully算法2.1.2 类Raft协议2.1.3 元数据合并 2.2 HA切换 3. 分片高可用3.1 集群分片汇报3.…

DoWhy:Python 中的因果推断库

DoWhy&#xff1a;Python 中的因果推断库 DoWhy 是一个强大的 Python 库&#xff0c;用于因果推断和因果推断分析。本文将介绍 DoWhy 的基本概念、主要功能和使用方法&#xff0c;帮助读者了解如何利用该库进行因果推断&#xff0c;并解决因果关系的相关问题。 什么是DoWhy&…

备战蓝桥杯---动态规划(理论基础)

目录 动态规划的概念&#xff1a; 解决多阶段决策过程最优化的一种方法 阶段&#xff1a; 状态&#xff1a; 决策&#xff1a; 策略&#xff1a; 状态转移方程&#xff1a; 适用的基本条件 1.具有相同的子问题 2.满足最优子结构 3.满足无后效性 动态规划的实现方式…

2024年【R2移动式压力容器充装】考试内容及R2移动式压力容器充装免费试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 R2移动式压力容器充装考试内容参考答案及R2移动式压力容器充装考试试题解析是安全生产模拟考试一点通题库老师及R2移动式压力容器充装操作证已考过的学员汇总&#xff0c;相对有效帮助R2移动式压力容器充装免费试题学…

【Java八股面试系列】JVM-内存区域

目录 Java内存区域 运行时数据区域 线程独享区域 程序计数器 Java 虚拟机栈 StackFlowError&OOM 本地方法栈 线程共享区域 堆 GCR-分代回收算法 字符串常量池 方法区 运行时常量池 HotSpot 虚拟机对象探秘 对象的创建 对象的内存布局 句柄 Java内存区域 运…

网络套件字(理论知识)

一、源IP地址和目的IP地址 上次说到IP地址是为了是为了让信息正确的从原主机传送到目的主机&#xff0c;而原IP地址和目的IP地址就是用于标识两个主机的&#xff0c;既然叫做地址必然有着路径规划的作用&#xff0c;而路径规划最重要的就是&#xff0c;从哪来到哪去&#xff0…

机器人学、机器视觉与控制 上机笔记(第一版译文版 2.1章节)

机器人学、机器视觉与控制 上机笔记&#xff08;第一版译文版 2.1章节&#xff09; 1、前言2、本篇内容3、代码记录3.1、新建se23.2、生成坐标系3.3、将T1表示的变换绘制3.4、完整绘制代码3.5、获取点*在坐标系1下的表示3.6、相对坐标获取完整代码 4、结语 1、前言 工作需要&a…

简单说网络:TCP+UDP

TCP和UPD: (1)都工作在传输层 (2)目的都是在程序之中传输数据 (3)数据可以是文本、视频或者图片(对TCP和UDP来说都是一堆二进制数没有太大区别) 一、区别:一个基于连接一个基于非连接 将人与人之间的通信比喻为进程和进程之前的通信:基本上有两种方式(1)写信;(2)打电话;这…

Docker容器化K8s集群部署教程(一键部署sheel脚本)

本文通过脚本&#xff0c;可以快速地部署和配置Kubernetes环境&#xff0c;省去了各插件手动部署、配置的繁琐过程。 先看最终结果&#xff1a; [rootlocalhost home]# kubectl get node NAME STATUS ROLES AGE VERSION k8smaster Ready control-p…

LlamaIndex 入门实战

文章目录 LlamaIndex 入门实战1. 基本概念2. 优劣势分析3. 简单代码示例4. Index持久化5. 使用场景6. 总结 LlamaIndex 入门实战 LlamaIndex是一个连接大型语言模型&#xff08;LLMs&#xff09;与外部数据的工具&#xff0c;它通过构建索引和提供查询接口&#xff0c;使得大模…

Java学习17-- super类

重点&#xff1a;super类 & 方法重写 super类 super指的是本级的上一级&#xff0c;即father class父类 很好理解&#xff0c;比如Person class>Student class 当前在Student class执行&#xff0c;那么就写this.xxx 需要在Student程序里面调用Person&#xff0c;那就…

HarmonyOS应用/服务发布:打造多设备生态的关键一步

目前 前言HarmonyOS 应用/服务发布的重要性使用HarmonyOS 构建跨设备的应用生态前期准备工作简述发布流程生成签名文件配置签名信息编译构建.app文件上架.app文件到AGC结束语 前言 随着智能设备的快速普及和多样化&#xff0c;以及编程语言的迅猛发展&#xff0c;构建一个无缝…

大数据 - Spark系列《四》- Spark分布式运行原理

Spark系列文章&#xff1a; 大数据 - Spark系列《一》- 从Hadoop到Spark&#xff1a;大数据计算引擎的演进-CSDN博客 大数据 - Spark系列《二》- 关于Spark在Idea中的一些常用配置-CSDN博客 大数据 - Spark系列《三》- 加载各种数据源创建RDD-CSDN博客 目录 &#x1f360;…

sqli-labs-master靶场训练笔记(38-53|boss战)

2024.2.4 level-38 &#xff08;堆叠注入&#xff09; 这题乍一看感觉又是来卖萌的&#xff0c;这不是和level-1一模一样吗 然后仔细看了一下源代码&#xff0c;根据 mysqli_multi_query 猜测这题的本意应该是堆叠注入 mysqli_multi_query() 是 PHP 中用于执行多个 SQL 查…

品牌如何营造生活感氛围?媒介盒子分享

「生活感」简而言之是指人们对生活的感受和意义&#xff0c;它往往没有充斥在各种重要的场合和事件中&#xff0c;而是更隐藏在细碎平凡的生活场景中。在营销越来越同质化的当下&#xff0c;品牌应该如何打破常规模式&#xff0c;洞察消费情绪&#xff0c;找到更能打动消费者心…

2023:AI疯狂进化年

嘿&#xff0c;大家好&#xff01;让我们一起来回顾一下这疯狂的 2023 年吧&#xff01;记得那个二月初吗&#xff1f;ChatGPT 上线了&#xff0c;然后呢&#xff1f;短短两个月&#xff0c;用户数量就像火箭一样突破了 1 亿&#xff01;这速度&#xff0c;简直比超级赛亚人还快…

前端JavaScript篇之对执行上下文的理解

目录 对执行上下文的理解创建执行上下文 对执行上下文的理解 当我们在执行JavaScript代码时&#xff0c;JavaScript引擎会创建并维护一个执行上下文栈来管理执行上下文。执行上下文有三种类型&#xff1a;全局执行上下文、函数执行上下文和eval函数执行上下文。 在写代码的时…

(6)【Python/机器学习/深度学习】Machine-Learning模型与算法应用—使用Adaboost建模及工作环境下的数据分析整理

目录 一、为什么要使用Adaboost建模? 二、泰坦尼克号分析(工作环境) (插曲)Python可以引入任何图形及图形可视化工具 三、数据分析 四、模型建立 1、RandomForestRegressor预测年龄 2、LogisticRegression建模 引入GridSearchCV 引入RandomizedSearchCV 3、Deci…