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

相关文章

在java中如何将inputStream对象转换为File对象(不生成本地文件)

《在java中如何将inputStream对象转换为File对象(不生成本地文件)》:本文主要介绍在java中如何将inputStream对象转换为File对象(不生成本地文件),具有很好的参考价... 目录需求说明问题解决总结需求说明在后端中通过POI生成Excel文件流,将输出流(outputStre

Linux中的进程间通信之匿名管道解读

《Linux中的进程间通信之匿名管道解读》:本文主要介绍Linux中的进程间通信之匿名管道解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、基本概念二、管道1、温故知新2、实现方式3、匿名管道(一)管道中的四种情况(二)管道的特性总结一、基本概念我们知道多

Qt 中 isHidden 和 isVisible 的区别与使用小结

《Qt中isHidden和isVisible的区别与使用小结》Qt中的isHidden()和isVisible()方法都用于查询组件显示或隐藏状态,然而,它们有很大的区别,了解它们对于正确操... 目录1. 基础概念2. 区别清见3. 实际案例4. 注意事项5. 总结1. 基础概念Qt 中的 isHidd

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

QT移植到RK3568开发板的方法步骤

《QT移植到RK3568开发板的方法步骤》本文主要介绍了QT移植到RK3568开发板的方法步骤,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录前言一、获取SDK1. 安装依赖2. 获取SDK资源包3. SDK工程目录介绍4. 获取补丁包二

Qt把文件夹从A移动到B的实现示例

《Qt把文件夹从A移动到B的实现示例》本文主要介绍了Qt把文件夹从A移动到B的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录如何移动一个文件? 如何移动文件夹(包含里面的全部内容):如何删除文件夹:QT 文件复制,移动(

Java实现将byte[]转换为File对象

《Java实现将byte[]转换为File对象》这篇文章将通过一个简单的例子为大家演示Java如何实现byte[]转换为File对象,并将其上传到外部服务器,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言1. 问题背景2. 环境准备3. 实现步骤3.1 从 URL 获取图片字节数据3.2 将字节数组

Javascript访问Promise对象返回值的操作方法

《Javascript访问Promise对象返回值的操作方法》这篇文章介绍了如何在JavaScript中使用Promise对象来处理异步操作,通过使用fetch()方法和Promise对象,我们可以从... 目录在Javascript中,什么是Promise1- then() 链式操作2- 在之后的代码中使

MyBatis的配置对象Configuration作用及说明

《MyBatis的配置对象Configuration作用及说明》MyBatis的Configuration对象是MyBatis的核心配置对象,它包含了MyBatis运行时所需的几乎所有配置信息,这个对... 目录MyBATis配置对象Configuration作用Configuration 对象的主要作用C

SpringBoot实现导出复杂对象到Excel文件

《SpringBoot实现导出复杂对象到Excel文件》这篇文章主要为大家详细介绍了如何使用Hutool和EasyExcel两种方式来实现在SpringBoot项目中导出复杂对象到Excel文件,需要... 在Spring Boot项目中导出复杂对象到Excel文件,可以利用Hutool或EasyExcel