Qt 之元对象

元对象(meta object)意思是描述另一个对象结构的对象,比如获得一个对象有多少成员函数,有哪些属性。在Qt中,我们将要用到的是QMetaObject这个类。

元对象系统基于以下3点:

  • 以QObject作为基类;
  • 类声明的私有区域中,Q_Object宏指令使我们能够使用元对象的特性,比如动态属性、信号、槽等;
  • 元对象编译器(Meta-Object Compiler  moc)为QObject子类生成具有元对象特性的代码

我们可以通过QObject类的一个成员函数获得该类的元对象:

QMetaObject *QObject::metaObject() const

通过这个元对象,进而可以获取一个QObject对象的更多信息:

返回运行时类的名称(不需要C++中的运行时类型识别机制RTTI)

QMetaObject::className() 

 返回类中方法的个数 

QMetaObject::methodCount() 

以上只是元对象的简单介绍,记住元对象系统的3点特性。之所以要介绍元对象,因为Qt中很多用法是基于元对象的,如果不支持元对象,比如没有继承自QObject,那么很多东西将无法使用

类型识别 

众所周知,C++中使用dynamic_cast和typeid这两个运算符进行运行时类型识别(RTII),但是Qt提供另外两种运行时类型识别方法:

qobject_cast 和 QObject::inherits()

看名字就可以知道,这两个方法都是基于QObject的,也就是元对象系统。

if (QLabel *label = qobject_cast<QLabel *>(obj))
{label->setText(tr("Ping"));
}
else if (QPushButton *button = qobject_cast<QPushButton *>(obj))
{button->setText(tr("Pong!"));
}

qobject_cast

  • qobject_cast 是 Qt 提供的一个宏,专门用于在 QObject 层次结构中进行安全的向下转型。
  • 它仅适用于继承自 QObject 的类,主要用于在使用 Qt 的对象树时,通过指针进行类型转换。
  • 如果对象类型不匹配,qobject_cast 返回 nullptr

dynamic_cast

  • dynamic_cast 是 C++ 的运行时类型识别(RTTI)操作符,用于在 C++ 的继承层次结构中执行类型安全的向下转型。
  • 它对于所有的 C++ 类型都有效,不仅仅是 QObject 的派生类。
  • 如果类型不匹配,dynamic_cast 返回 nullptr(对于指针)或抛出 std::bad_cast 异常(对于引用)。

QObject::inherits(const char *className)的速度相对慢一些,所以尽可能使用qobject_cast。

Qt中的属性

1. 自定义属性

我们可能已经接触到很多Qt中的属性了,比如qreal类型的opacity属性表示“透明度”,QRect类型的geometry表示“几何位置和大小”,QPoint类型的pos属性代表“位置”。所谓属性,也就是类中的一个数据成员,我们可以获取(get)和设置(set)。除了Qt中一些类已经具备的属性,我们还可以自定义属性,也就是定义一种访问数据成员的方式。

在一个继承自QObject的类中使用 Q_PROPERTY 宏指令,比如:

Q_PROPERTY(bool focus READ hasFocus)Q_PROPERTY(bool enabled READ isEnabled WRITE setEnabled)Q_PROPERTY(QColor color MEMBER m_color NOTIFY colorChanged)

不要被这个用法搞晕了,其实很简单,刚开始指定属性类型和名称,然后READ表示获取属性值的方法,一般这两点是必须的。其他都是可选的,比如WRITE表示设置属性值得方法,MEMBER表示这个属性在类中数据成员的名称NOTIFY表示属性改变发出的信号

2. 属性的类型

由上可知,属性的类型可以是bool、QString、QRect等等,我们可以通过 QVariant::Type 的枚举值获得所有可用于属性的类型。可以查到,它不支持枚举类型,但可以通过Q_ENUM来设置:

enum Priority { High, Low, VeryHigh, VeryLow };Q_ENUM(Priority)Q_PROPERTY(Priority priority READ priority WRITE setPriority NOTIFY priorityChanged)

当然,自定义的类型也是支持的,需要通过Q_DECLARE_METATYPE 注册元类型:

struct MyStruct{int i;...};Q_DECLARE_METATYPE(MyStruct)

3. 属性的读与写

我们可以直接使用get和set方法来读写属性,也可以通过QObject与QMetaObject来间接地读写属性。

首先是设置属性值:

比如类QAbstractButton有一个“down”的属性,表示按钮是否被按下,它有一个成员函数 QAbstractButton::setDown() 来改变属性值,同时,我们也可以通过 QObject::setProperty() 对其进行设置:

QPushButton *button = new QPushButton;QObject *object = button;button->setDown(true);object->setProperty("down", true);

值得注意的是,setProperty()这个函数不但可以改变属性值,也可以在运行时动态地为对象添加属性。

接下来是读取属性值:

如果有get函数,可以直接调用它,当然也可以通过 QObject::property() 来获取属性,它的返回值是 QVariant 类型的,通过 canConvert() 进行判断,然后将其转换为所需的类型。

QObject *object = ...const QMetaObject *metaobject = object->metaObject();int count = metaobject->propertyCount();for (int i=0; i<count; ++i) {QMetaProperty metaproperty = metaobject->property(i);const char *name = metaproperty.name();QVariant value = object->property(name);...}

invokeMethod()


Qt中的信号槽机制是以元对象为基础的,通过名称以类型安全的方式来间接调用槽函数。

当调用槽函数时,实际是由invokeMethod()完成的。

比如显示一个窗口,一般是通过show()函数来完成,不过我们还能这样做:

MyWidget w;QMetaObject::invokeMethod(&w, "show");

上面讲了如何将成员变量注册进元对象系统,那么对于成员函数,该怎么做呢?

在声明一个类的成员函数时,通过使用 Q_INVOKABLE 宏进行注册,可以使它们能够被元对象系统调用。

class Window : public QWidget{Q_OBJECTpublic:Window();void normalMethod();Q_INVOKABLE void invokableMethod();};

反射机制

反射机制的特性


    reflection 模式(反射模式或反射机制):是指在运行时,能获取任意一个类对象的所有类型信息、属性、成员函数等信息的一种机制。

元对象系统提供的功能之一是为 QObject 派生类对象提供运行时的类型信息及数据成员的当前值等信息,也就是说,在运行阶段,程序可以获取 QObject 派生类对象所属类的名称、父类名称、该对象的成员函数、枚举类型、数据成员等信息,其实这就是反射机制。

因为 Qt 的元对象系统必须从 QObject 继承,又从反射机制的主要作用可看到,Qt 的元对象系统主要是为程序提供了 QObject 类对象及其派生类对象的信息,也就是说不是从 QObject 派生的类对象,则无法使用 Qt 的元对象系统来获取这些信息。

 Qt 具体实现反射机制的方法

Qt 使用了一系列的类来实现反射机制,这些类对对象的各个方面进行了描述,其中QMetaObject 类描述了 QObject 及其派生类对象的所有元信息,该类是 Qt 元对象系统的核心类,通过该类的成员函数可以获取 QObject 及其派生类对象的所有元信息,因此可以说 QMetaObject 类的对象是 Qt 中的元对象。注意:要调用 QMetaObject 类中的成员函数需要使用 QMetaObject 类型的对象。

对对象的成员进行描述:一个对象包含数据成员、函数成员、构造函数、枚举成员等成员,在 Qt 中,这些成员分别使用了不同的类对其进行描述,比如函数成员使用类QMetaMethod 进行描述,属性使用 QMetaProperty 类进行描述等,然后使用QMetaObject 类对整个类对象进行描述,比如要获取成员函数的函数名,其代码如下:

QMetaMethod qm = metaObject->method(1); //获取索引为 1 的成员函数
qDebug()<<qm.name()<<"\n"; //输出该成员函数的名称。

信号和槽函数的实现原理

Qt中信号和槽的实现原理基于元对象系统(Meta-Object System,MOS)。以下是信号和槽的实现原理:

  1. 元对象系统: 在包含信号和槽的类声明中,使用 Q_OBJECT 宏。这个宏会告诉 Qt 的元对象编译器(MOC)处理这个类,生成额外的代码。这些额外的代码包括元对象的描述信息,以及支持信号和槽机制的相关数据。

  2. 元对象: Qt 的元对象是对类的额外描述,它包含了类的属性、方法、信号和槽的信息。这些信息被存储在元对象表中。

  3. 信号和槽的注册: 在包含信号和槽的类的构造函数中,MOC 自动生成的代码会注册这些信号和槽。这个注册过程将信号和槽的信息添加到元对象表中。

  4. 连接: 使用 connect 函数建立信号和槽之间的连接。在连接时,Qt 运行时系统会查找元对象表,找到信号和槽的信息。这时,建立了信号和槽之间的关联。

  5. emit关键字: 当使用 emit 关键字发射一个信号时,实际上是调用了生成的槽函数。这个槽函数内部会遍历连接列表,调用所有连接到该信号的槽函数。

  6. 运行时调用: Qt 使用元对象系统在运行时实现信号和槽的调用。这样,它允许在不知道类的具体实现的情况下,进行动态的信号和槽的连接和调用。

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

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

相关文章

基于蚁狮算法优化概率神经网络PNN的分类预测 - 附代码

基于蚁狮算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于蚁狮算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于蚁狮优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要&#xff1a;针对PNN神经网络的光滑…

【Leetcode Sheet】Weekly Practice 15

Leetcode Test 2586 统计范围内的元音字符串数(11.7) 给你一个下标从 0 开始的字符串数组 words 和两个整数&#xff1a;left 和 right 。 如果字符串以元音字母开头并以元音字母结尾&#xff0c;那么该字符串就是一个 元音字符串 &#xff0c;其中元音字母是 a、e、i、o、u…

Golang 整合RocketMQ

RocketMQ 相关知识汇总 RocketMQ 是什么 RocketMQ 是阿里巴巴开源的一款 MQ 框架&#xff0c;被广泛的使用于不同的业务场景&#xff0c;同时也有非常好的生态系统支持&#xff0c;支持事务消息、顺序消息、批量消息、定时消息、消息回溯等功能。 RocketMQ核心概念 名称服务(N…

【已验证】php配置连接sql server中文乱码(解决方法)更改utf-8格式

解决数据库中的中文数据在页面显示乱码的问题 在连接的$connectionInfo中设置"CharacterSet" > "UTF-8"&#xff0c;指定编码方式即可 $connectionInfo array("UID">$uid, "PWD">$pwd, "Database">$database…

【springboot】Failed to start bean ‘webServerStartStop‘;

新同事新建了一个项目springboot项目&#xff0c;启动时候报错。 具体错误如下&#xff1a; Failed to start bean webServerStartStop; nested exception is org.springframework.boot.web.server.WebServerException: Unable to start embedded Tomcat server 未能启动bea…

(论文阅读26/100)Weakly-supervised learning with convolutional neural networks

26.文献阅读笔记 简介 题目 Weakly-supervised learning with convolutional neural networks 作者 Maxime Oquab&#xff0c;Leon Bottou&#xff0c;Ivan Laptev&#xff0c;Josef Sivic&#xff0c;CVPR&#xff0c;2015 原文链接 http://www.cv-foundation.org/open…

深度探究深度学习常见数据类型INT8 FP32 FP16的区别即优缺点

定点和浮点都是数值的表示&#xff08;representation&#xff09;&#xff0c;它们区别在于&#xff0c;将整数&#xff08;integer&#xff09;部分和小数&#xff08;fractional&#xff09;部分分开的点&#xff0c;点在哪里。定点保留特定位数整数和小数&#xff0c;而浮点…

C++——const成员

这里先用队列举例&#xff1a; #define _CRT_SECURE_NO_WARNINGS 1 #include <iostream> #include <assert.h> using namespace std; class SeqList { public:void pushBack(int data){if (_size _capacity){int* tmp (int*)realloc(a, sizeof(int) * 4);if (tm…

excel记录wFm数值(推理过程)

1 导入计算wfm库2 实例化具体的指标 3 列表循环之前&#xff0c;设置空list 4 单图评测-将图号、图片名、数值记录 列表里面存储dict 5 将excel列表结果逐个存入excel.xlsx文件 完整代码 test_CPD.py ### test_CPD.py ### import torch import torch.nn.functional as Fimpor…

算法leetcode|88. 合并两个有序数组(rust重拳出击)

文章目录 88. 合并两个有序数组&#xff1a;样例 1&#xff1a;样例 2&#xff1a;样例 3&#xff1a;提示&#xff1a; 分析&#xff1a;题解&#xff1a;rust&#xff1a;go&#xff1a;c&#xff1a;python&#xff1a;java&#xff1a; 88. 合并两个有序数组&#xff1a; …

flv.js在vue中的使用

Flv.js 是 HTML5 Flash 视频&#xff08;FLV&#xff09;播放器&#xff0c;纯原生 JavaScript 开发&#xff0c;没有用到 Flash。由 bilibili 网站开源。它的工作原理是将 FLV 文件流转码复用成 ISO BMFF&#xff08;MP4 碎片&#xff09;片段&#xff0c;然后通过 Media Sour…

​《水经注全国三维离线GIS系统》硬件安装教程

有些工作&#xff0c;是需要一些外在动力才能完成的。 为什么这么讲呢&#xff1f; 因为正是在客户的要求下&#xff0c;我们才撰写了《水经注全国三维离线GIS系统》的硬件安装教程&#xff0c;而且还录制了视频教程。 当用户收到货物以后&#xff0c;就可以通过本教程清点货…

信驰达科技加入车联网联盟(CCC),推进数字钥匙发展与应用

CCC)的会员。 图 1 深圳信驰达正式成为车联网联盟(CCC)会员 车联网联盟(CCC)是一个跨行业组织&#xff0c;致力于推动智能手机与汽车连接解决方案的技术发展。CCC涵盖了全球汽车和智能手机行业的大部分企业&#xff0c;拥有150多家成员公司。CCC成员公司包括智能手机和汽车制造…

chose_xml

import os import shutil # 定义函数&#xff0c;用于遍历文件夹并复制文件 def copy_files(src_folder, dst_folder, file_type): # 遍历文件夹 for root, dirs, files in os.walk(src_folder): # 遍历文件 for file in files: # 判断文…

Springboot+vue的人力资源管理系统(有报告)。Javaee项目,springboot vue前后端分离项目

演示视频&#xff1a; Springbootvue的人力资源管理系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot vue前后端分离项目 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的企业资产管理系统&#xff0c;采用M&#xff08;model&…

react+星火大模型,构建上下文ai问答页面(可扩展)

前言 最近写的开源项目核心功能跑通了&#xff0c;前两天突发奇想。关于项目可否介入大模型来辅助用户使用平台&#xff0c;就跑去研究了最近比较活火的国内大模型–讯飞星火大模型。 大模型api获取 控制台登录 地址&#xff1a;https://console.xfyun.cn/app/myapp 新建应…

迅为龙芯2K1000开发板虚拟机ubuntu启动root用户

作为嵌入式开发人员&#xff0c;系统的所有权限都要为我们打开&#xff0c;所以我们不必像运维那样&#xff0c;对 root 用户非常敏感&#xff0c;所以安装完 ubuntu 系统以后&#xff0c;我们要启用 root 用户。 首先我们打开 ubuntu 控制终端&#xff0c;然后在终端里面输入…

[SOC] MBIST (Memory Built-In Self Test) and Memory Built-in Self Repair (BISR)

存储器构成了 VLSI 电路的很大一部分。存储系统设计的目的 是存储大量数据。[1] 存储器不包括逻辑门和触发器。因此&#xff0c;需要不同的故障模型和测试算法来测试存储器。 MBIST 是一种自测试和修复机制&#xff0c;它通过一组有效的算法来测试存储器&#xff0c;以检测典型…

【阿里云】任务2-OSS对象存储教程(找我参加活动可获得京东卡奖励)

目录 前言说明第一步第二步第三步&#xff1a;开通并使用OSS传输加速三、清理第四步-提交作品第五步-提交记录到小程序 前言 本次任务是阿里云官方发出的&#xff0c;每个任务30软妹币&#xff0c;欢迎大家加入我的活动群&#xff0c;门槛很低&#xff0c;所有人都可以参加&…

代码随想录算法训练营第五十天丨 动态规划part13

300.最长递增子序列 思路 首先通过本题大家要明确什么是子序列&#xff0c;“子序列是由数组派生而来的序列&#xff0c;删除&#xff08;或不删除&#xff09;数组中的元素而不改变其余元素的顺序”。 本题也是代码随想录中子序列问题的第一题&#xff0c;如果没接触过这种…