深入Qt 学习 -- 反射机制(比较简单清楚)
相对于Java天生的这一特性, C++并不具备;但进入到Qt领域,这一切都变得简单自如了。
从Qt的元对象系统可知,除了提供信号/槽机制的特性之外,它还提供了以下特性:
|
■ QObject::metaObject() |
返回关联的元对象 |
|
■ QObject::className() |
在运行时状态下返回类名 |
|
■ QObject::inherits() |
判断类的继承关系 |
|
■ QObject::tr() QObject::trUtf8() |
提供国际化,翻译字符串 |
|
■ QObject::setProperty() QObject::property() |
通过名称来动态设置和获取属性 |
|
■ QObject::newInstance() |
创建实例 |
其中QObject::className()、QObject::setProperty()和QObject::property()比较吸引眼球,这正是反射机制中的基本功能。建立在Qt的元对象系统的基础上,只要拥有元对象(QMetaObject)的类就支持反射!如何拥有元对象?方法很简单:只需继承于QObject或者它的子类(多重继承的话,QObject类应该放在第一个),在头文件的结构声明中写上Q_OBJECT即可!
元对象信息
通过QObject::metaObject()方法, 所有继承于QObject的类可以 返回元对象系统为其生成的metaObject对象。看看QMetaObject提供的一些重要信息:
|
■ QMetaClassInfo |
通过宏Q_CLASSINFO的支持,提供类的附加信息 |
|
■ QMetaEnum |
Qt特色的枚举对象,支持key和 value之间的互转 |
|
■ QMetaMethod |
提供类成员函数的元数据 |
|
■ QMetaProperty |
提供类成员属性的元数据 |
根据QMetaObject提供的数据对象,完全可以编写通用的代码来支持反射特性。
例子:
声明一个类,继承于QObject
|
classReflectionObject : publicQObject { Q_OBJECT Q_PROPERTY(intId READ Id WRITE setId) Q_PROPERTY(QString Name READ Name WRITEsetName) Q_PROPERTY(QString Address READ Address WRITEsetAddress) Q_PROPERTY(PriorityType Level READ Priority WRITEsetPriority) Q_ENUMS(PriorityType) public: enumPriorityType { High, Low, VeryHigh,VeryLow }; Q_INVOKABLEint Id() {returnm_Id; } Q_INVOKABLEQString Name() { returnm_Name; } Q_INVOKABLEQString Address() { returnm_Address; } Q_INVOKABLEPriorityType Priority() const {returnm_Priority; } Q_INVOKABLEvoid setId(constint& id) {m_Id = id; } Q_INVOKABLEvoid setName(constQString& name) {m_Name = name; } Q_INVOKABLEvoid setAddress(constQString& address) {m_Address = address; } Q_INVOKABLEvoid setPriority(PriorityType priority) {m_Priority = priority; } private: int m_Id; QString m_Name; QString m_Address; PriorityTypem_Priority; }; |
为了能检测到类成员函数,得在函数前加上一个宏Q_INVOKABLE, 这意味着该函数在元对象系统编译该类时注册该函数,则在运行过程中能被元对象调用。
经过上述声明,在运行时即可做些操作:
/*遍历该类的成员: */
|
ReflectionObjecttheObject; constQMetaObject*theMetaObject =theObject.metaObject(); intmetathodIndex; intmetathodCount = theMetaObject->methodCount(); for(metathodIndex = 0; metathodIndex < metathodCount; ++metathodIndex) { QMetaMethodoneMethod =theMetaObject->method(metathodIndex); qDebug() <<"typeName: " <<oneMethod.typeName(); qDebug() <<"signature: " <<oneMethod.signature(); qDebug() <<"methodType: " <<oneMethod.methodType();; qDebug() <<"parameterNames: " <<oneMethod.parameterNames() <<"\n"; } |
/*遍历该类的属性: */
|
intpropertyIndex; intpropertyCount = theMetaObject->propertyCount(); for(propertyIndex = 0; propertyIndex < propertyCount; ++propertyIndex) { QMetaPropertyoneProperty =theMetaObject->property(propertyIndex); qDebug() <<"name: " << oneProperty.name(); qDebug() <<"type: " <<oneProperty.type() <<"\n"; } |
/*遍历该类的枚举集合*/
|
intenumeratorIndex; intenumeratorCount = theMetaObject->enumeratorCount(); for(enumeratorIndex = 0; enumeratorIndex < enumeratorCount; ++enumeratorIndex) { QMetaEnumoneEnumerator =theMetaObject->enumerator(enumeratorIndex); intenumIndex; intenumCount = oneEnumerator.keyCount(); for(enumIndex = 0;enumIndex < enumCount; ++enumIndex) { qDebug() <<oneEnumerator.key(enumIndex) <<" ==> " <<oneEnumerator.value(enumIndex); } } |
在这里我们看到QMetaEnum存在key、value配对出现,这必然可以互转,而QMetaEnum确实提供了方式:valueToKey()、keyToValue()。
通过这个测试能将ReflectionObject这个类的方法和属性完全遍历出来,完成自身的检查,也即反射。
反射在Qt应用程序中的用途
在Qt的元对象系统支持下,赋予了C++并不直接拥有的此特性。这样加大了开发应用程序的自由度,尤其在软件工程中强调高内聚低耦合的状态下。
具体的用例中, 可以通过Qt Designer这个工具来描述(尽管暂不清楚内部是否利用此方法来实现J):
对于其中的控件,放入到Designer中时,Designer事先并不知道(其实也不需要知道)放入插件的类型、方法和属性。将插件导入时,它可以检测该控件的类型、方法和属性;
这样在Designer中绘制控件时,遍历该控件的属性, 将这些属性显示在属性窗口中,便于修改;
在信号/槽对话框编辑时,遍历该控件的属性,提取信号/槽函数,放入对应的编辑区域里;
http://blog.csdn.net/playstudy/article/details/7861329
深入Qt 学习 -- 反射机制(比较简单清楚)的更多相关文章
- (转)java反射机制及简单工厂模式
第一步,定义一个接口类 package factory.face; /** * 接口,为了保证返回的对象可以统一用 Product接受 * @author Administrator */ publi ...
- java反射机制的简单介绍
参考博客: https://blog.csdn.net/mlc1218559742/article/details/52754310 先给出反射机制中常用的几个方法: Class.forName (& ...
- 045-利用反射机制,简单的实现PHP插件模式
<?php //利用反射机制,简单的实现PHP插件模式 # 假设,我们有一款开源产品,所有开发者都必须在我定制的需求之上,进行二次开发, # 而开发完成后的新模块,就是一个不一样的新插件,可以放 ...
- Java反射机制的简单学习
今天看了一下Java的反射机制,就此记录一下. 首先,我们要先了解一下什么是反射? 反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力. JAV ...
- Qt类反射机制
前言 如何通过类名动态的创建对象(类反射),对于Java开发者来说一点也不陌生,当然Qt也提供了一个元对象系统(The Meta-Object System),通过这个系统我们可以动态的在运行时获 ...
- 利用Java反射机制优化简单工厂设计模式
之前项目有个需求,审批流程的时候要根据配置发送信息:发送短信.发送邮件.当时看到这个就想到要用工厂模式,为什么要用工厂模式呢?用工厂模式进行大型项目的开发,可以很好的进行项目并行开发.就是一个程序员和 ...
- C++ 反射机制的简单实现
C++并不支持反射机制,只能自己实现. 如果需要实现字字符串到函数到映射,一定要使用到函数指针. 简单实现反射机制,根据字符串来构造相应到类.主要有以下几点: (1) 可以使用map保存字符从到函数指 ...
- Java反射机制的简单应用
一直感觉java的反射机制非常强大,可是可用的地方不多.在android学习的时候.一直想实现挂断电话的功能,可是系统并没有提供开放的api接口,看了一下网上使用反射机制来实现该功能,确实非常强大,非 ...
- java学习——反射机制
/* * JAVA反射机制是在运行状态中,对于任意一个类 (class文件),都能够知道这个类的所有属性和方法: * 对于任意一个对象,都能够调用它的任意一个方法和属性: * 这种动态获取的信息以及动 ...
随机推荐
- android请求
//请求 HttpURLConnection conn = (HttpURLConnection)new URL(path).openConnection(); conn.setConnecTimeo ...
- Vivado 2017.2 SDK 生成FSBL时存在的bug
SDK 2017.1/.2 - ld.exe: cannot find -lrsa When importing a new HDF file into the SDK or after a clea ...
- [NIO]用dawn发送接收HTTP请求
HTTP协议的下层使用的是tcp.所以我们建立一个tcp连接就能发送接收http请求.dawn底层使用了nio.可是经过dawn的封装之后,我们在编写代码的时候,就和使用普通的堵塞式socket一样 ...
- 基于Android Fragment功能的样例
通过近期空暇时候对Fragment的学习,尝试着写了一个小Demo,将在开发的时候能经常使用到的Fragment知识放在一起,写过了这个Demo对Android Fragment的了解更加深入了,以后 ...
- 我的前端规范——HTML篇
相关文章 简书原文:https://www.jianshu.com/p/a46ff0504982 我的前端规范——开篇:http://www.cnblogs.com/shcrk/p/9271561.h ...
- ActiveMQ简单入门实例
一.下载MQ 官方网站下载:http://activemq.apache.org/ 我用的是 apache-activemq-5.15.0-bin 二.安装 我用的是64位所以双击 apache-ac ...
- Android Wear之android穿戴式设备应用开发平台
Android Wear于2014年03月19日公布,并有Moto 360和LG watch两款产品. 眼下源代码还没有开放.可是开发人员能够下载它的Image及相应的开发SDK,这样开发人员通过模拟 ...
- Finder那点事
事件是这样,我MAC PRO,关不了机了,是有什么线程在用 defaults write com.apple.Finder QuitMenuItem 1 这个命令是让Finder 有退出BTN ,co ...
- [Compose] 12. Two rules about Funtors
We learn the formal definition of a functor and look at the laws they obey. Any Functor should follo ...
- synology
入手群晖261J无法正常安装DSM 错误代码38 求教各位恶魔https://www.chiphell.com/thread-1599081-1-1.html(出处: Chiphell - 分享与交流 ...