QT5.14.2 探秘Qt信号槽奥秘--让对象间通信如虎添翼

2024-03-19 16:12

本文主要是介绍QT5.14.2 探秘Qt信号槽奥秘--让对象间通信如虎添翼,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


一、前言


在当今这个万物互联的时代,对象间通信无疑是编程领域中最为基础也最为重要的问题。作为知名的跨平台开发框架,Qt自然也需要解决这一问题。于是,Qt巧妙地提出了信号与槽(Signals & Slots)这一机制,以观察者模式的思路让对象间通信变得行云流水。那么,Qt信号与槽的本质是什么?它是如何在底层实现的?又有哪些实战应用技巧?本文将为您一一道来。


二、Qt信号槽本质剖析


1、Qt信号槽的核心思想

  • 信号(Signal)作为对象状态或行为变化的通知;

  • 槽(Slot)作为相应该通知的处理函数;

  • 通过Connect将信号与槽关联,使通知能精准触达指定槽函数。

从语法上看,Qt通过自创的signals、slots和emit等关键字模拟了信号与槽:


signals: // 声明信号void valueChanged(int newValue);public slots: //声明槽函数void valueChangedHandler(int value); //发射信号 
emit valueChanged(25);

2、信号和槽的语法规则

  • 信号必须用signals关键字声明在类定义中

  • 信号没有函数体实现

  • 槽可以是普通的成员函数,或用slots关键字声明

  • 只有继承自QObject的类才能定义信号和槽

  • 只有在QObject子类实例化后才能使用信号和槽机制


3、信号与槽之间的关系

在这里插入图片描述


通过这些规则和关系,Qt构建了一个高度灵活可扩展的信号与槽通信机制。


三、信号槽原理揭秘

那么信号和槽在底层又是如何实现的呢?原理如下:

  1. Qt的moc(Meta-Object Compiler元对象编译器)会扫描通过Q_OBJECT宏声明的类,为其生成元对象数据(elements/meta-object.cpp等)

  2. 该元数据包含了信号与槽的名称、参数等完整信息

  3. 对象在运行时,根据这些元数据查找并调用目标槽函数

  4. 通过QObject::connect()函数将信号与槽关联起来:


QObject::connect(sender, signal, receiver, slot)

其中:

  • sender是发送信号的对象
  • signal是发送的信号,使用&Sender_Class::signal语法指定
  • receiver是接收信号的对象
  • slot是要调用的槽函数,使用&Receiver_Class::slot语法指定

Connect()函数一共有多种重载形式,支持使用函数指针或者字符串的方式指定信号和槽。

让我们看一个实例:

// 定义发送者和接收者对象
QPushButton *button = new QPushButton;
QDialog *dialog = new QDialog;// 使用函数指针方式连接信号与槽
QObject::connect(button, &QPushButton::clicked,dialog, &QDialog::accept);// 使用字符串方式连接信号与槽
QObject::connect(button, SIGNAL(clicked()),dialog, SLOT(accept()));

上例中,当按钮被点击时,就会发射clicked()信号,进而触发QDialog的accept()槽函数被调用。

需要注意的是,使用函数指针的方式更加安全,支持参数隐式转换以及自定义类型,而字符串方式则需要确保参数名称和顺序完全一致。

由此可见,QObject::connect()是将发送者对象的信号绑定到接收者对象的槽函数上,从而建立两者间的通信通道。这正是Qt信号与槽大显身手的核心所在。

因此,Qt信号槽的本质就是一种基于元数据的动态绑定机制。相比函数指针,它更加智能且自动化。这种设计使得Qt极大提升了多态、可扩展性等面向对象特性,成为当代C++大作的有力支撑。


四、代码实战:订单监控器

理论说了那么多,我们还是通过一个实战案例来印证一下信号槽的魔力。

比如我们需要开发一款订单状态监控器:当订单状态发生变化时,自动记录日志、发送通知等。

这种监听机制,正好可以利用信号槽来实现:

//订单类
class Order : public QObject
{Q_OBJECTpublic:enum Status {Open, Resolved, Closed};Order(QObject *parent = nullptr): QObject(parent), m_status(Open){}void setStatus(Status status){if (status != m_status) {m_status = status;emit statusChanged(status); //发射信号}}signals:void statusChanged(Order::Status newStatus);private:Status m_status;
};//监控器类
class Monitor : public QObject
{Q_OBJECTpublic:Monitor(QObject *parent = nullptr): QObject(parent){Order *order = new Order(this);connect(order, &Order::statusChanged, //连接信号与槽this, &Monitor::logStatus);order->setStatus(Order::Resolved); //状态变化,触发信号order->setStatus(Order::Closed); }private slots:void logStatus(Order::Status status){static const char *statusStr[] = {"Order Open","Order Resolved","Order Closed"};std::cout << statusStr[status] << std::endl;}
};

在上面的示例中,我们首先定义了Order类表示订单,它有一个statusChanged信号,在订单状态变化时会发射该信号。

然后定义了Monitor类作为监控器,在构造函数中:

  1. 创建Order对象
  2. 使用connect()函数连接Order对象的statusChanged信号与Monitor的logStatus槽函数
  3. 改变Order状态,触发信号发射,从而调用logStatus输出日志

其中,关键的连接代码是:

connect(order, &Order::statusChanged, this, &Monitor::logStatus);

这一语句就将order对象的statusChanged信号与this(Monitor实例)的logStatus槽绑定到了一起。

这样,只要订单状态改变,statusChanged信号就会被触发,Monitor的logStatus槽函数就会被自动调用,输出相应的日志,而不需要手动去调用。整个过程是自动完成的。

通过这个例子,我们可以看到Qt信号槽机制让对象间的监听与通信变得非常简单直观,大大增强了代码的可维护性和扩展性。


五、实战技巧剖析

通过上面的例子,我们可以总结出Qt信号槽的以下实战技巧:

  1. Qt分离了事件主体和监听器的耦合,增强了可维护性
  2. 信号槽允许参数传递,方便数据共享
  3. 一个信号可连接多个槽,一个槽也可被多个信号调用,扩展性强
  4. 信号可自动执行,不需要手动调用,代码更加简洁
  5. 使用lambda表达式连接信号槽,代码更紧凑
  6. 信号可过滤,根据条件选择是否执行槽
  7. Qt内置许多常用信号,方便快速集成

六、未来展望

本文从本质到实现,为您讲解了Qt信号槽的方方面面。相信通过这个机制,您对面向对象编程的理解会有进一步的提升。作为Qt框架的核心机制,信号与槽不仅让对象间通信变得行云流水,更是彰显了C++语言强大的可扩展性。通过Qt信号槽,我们可以写出更优雅、更易维护的面向对象代码。信号就是力量,与槽实现无限可能,期待您的大显身手!

这篇关于QT5.14.2 探秘Qt信号槽奥秘--让对象间通信如虎添翼的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/826557

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

Java第二阶段---09类和对象---第三节 构造方法

第三节 构造方法 1.概念 构造方法是一种特殊的方法,主要用于创建对象以及完成对象的属性初始化操作。构造方法不能被对象调用。 2.语法 //[]中内容可有可无 访问修饰符 类名([参数列表]){ } 3.示例 public class Car {     //车特征(属性)     public String name;//车名   可以直接拿来用 说明它有初始值     pu

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点

java线程深度解析(二)——线程互斥技术与线程间通信

http://blog.csdn.net/daybreak1209/article/details/51307679      在java多线程——线程同步问题中,对于多线程下程序启动时出现的线程安全问题的背景和初步解决方案已经有了详细的介绍。本文将再度深入解析对线程代码块和方法的同步控制和多线程间通信的实例。 一、再现多线程下安全问题 先看开启两条线程,分别按序打印字符串的

使用Qt编程QtNetwork无法使用

使用 VS 构建 Qt 项目时 QtNetwork 无法使用的问题 - 摘叶飞镖 - 博客园 (cnblogs.com) 另外,强烈建议在使用QNetworkAccessManager之前看看这篇文章: Qt 之 QNetworkAccessManager踏坑记录-CSDN博客 C++ Qt开发:QNetworkAccessManager网络接口组件 阅读目录 1.1 通用API函数

HTML5自定义属性对象Dataset

原文转自HTML5自定义属性对象Dataset简介 一、html5 自定义属性介绍 之前翻译的“你必须知道的28个HTML5特征、窍门和技术”一文中对于HTML5中自定义合法属性data-已经做过些介绍,就是在HTML5中我们可以使用data-前缀设置我们需要的自定义属性,来进行一些数据的存放,例如我们要在一个文字按钮上存放相对应的id: <a href="javascript:" d

列举你能想到的UNIX信号,并说明信号用途

信号是一种软中断,是一种处理异步事件的方法。一般来说,操作系统都支持许多信号。尤其是UNIX,比较重要应用程序一般都会处理信号。 UNIX定义了许多信号,比如SIGINT表示中断字符信号,也就是Ctrl+C的信号,SIGBUS表示硬件故障的信号;SIGCHLD表示子进程状态改变信号;SIGKILL表示终止程序运行的信号,等等。信号量编程是UNIX下非常重要的一种技术。 Unix信号量也可以

PHP7扩展开发之对象方式使用lib库

前言 上一篇文章,我们使用的是函数方式调用lib库。这篇文章我们将使用对象的方式调用lib库。调用代码如下: <?php $hello = new hello(); $result = $hello->get(); var_dump($result); ?> 我们将在扩展中实现hello类。hello类中将依赖lib库。 代码 基础代码 这个扩展,我们将在say扩展上增加相关代码。sa

Qt多语种开发教程

Qt作为跨平台的开发工具,早已应用到各行各业的软件开发中。 今天讲讲,Qt开发的正序怎么做多语言开发。就是说,你设置中文,就中文显示;设置英语就英文显示,设置繁体就繁体显示,设置发育就显示法语等。 开发环境(其实多语种这块根环境没太大关系):win10,Qt.5.12.10 一.先用QtCreator创建一个简单的桌面程序 1.工程就随便命名“LanguageTest”,其他默认。 2.在设计师