解析 Qt 内省机制
创始人
2024-08-02 08:31:49
0

本文介绍的是Qt 内省机制,关于内省,新手的原因,我们一块学习,所谓内省是指面向对象语言的一种在运行期间查询对象信息的能力, 比如如果该语具有运行期间检查对象型别的能力,那么我们称它是型别内省(type intropection)的,型别内省可以用来实施多态。

c++的内省比较有限,它仅支持上面所说的型别内省, C++的型别内省是通过运行时类型识别(RTTI)(Run-Time Type Information)中的typeid 以及 dynamic_case关键字来实现的,举例说明:

  1.     // rabbit 派生于 Animal, jump为虚函数  
  2.  
  3. if ( rabbit *p = dynamic_case(obj))     
  4. {     
  5.    p->jump();     
  6. }     
  7. //我们还可以通过typeid萃取到对象的型别信息,比如对象的名称   
  8. std::cout << typeid(obj).name() << std::endl   

Qt拓展了C++的内省机制,(实际上,它并没有采用c++的RTTI),而是提供了更为强大的元对象(meta object)机制,来实现内省。接下来,就让我们看看,Qt是如何扩展c++内省机制的。

要深刻理解Qt的内省机制,首先理解QObject,QObject类是整个Qt对象模型的心脏,Qt对象模型最为核心的功能是提供一种无缝的对象通讯机制,即就是我们所熟知的信号和槽。QObject主要有三大职责: 内存管理、内省(intropection)与事件处理。本文将集中在在内省的讨论。以下代码介绍了QObject类提供的内省方法: 

  1. //每个对象可以通过QObject::setObjectName()和QObject::objectName()设置、取得类的实例的名字   
  2. FirstQtApp obj;    
  3. obj.setObjectName("instanceName");    
  4. QString name1 = obj.objectName();   // return instanceName   
  5. //每个对象还可以通过它的元对象className方法得到类的名字   
  6. QString name2 = obj.metaObject()->className();  // return FirtstQtApp   
  7. //每个对象可以通过QObject::inherits方法来查询是否对前对象类派生于量一个类   
  8. bool isherited =  obj.inherits("QObject");         // returns true   
  9. isherited =  obj.inherits("QWideget");         // returns true  

让我们再来一下QObject::inherits方法的底层实现:

  1. inline bool inherits(const char *classname) const   
  2.     { return const_cast(this)->qt_metacast(classname) != 0; }  

原来,QObject::inherits是通过qt_metacast()这个虚函数实现的, 事实上每个QObject的派生类都必须实现metaObject()以及其他qt_metacall()方法,从而满足自省方法className, inherits等方法的调用(当然还有其他用途)。

而所有有关派生从QObject的子类中的内省方法无须有用户实现,用户只要在类中声明宏Q_OBJECT即可,Qt的元对象编译器(moc)负责实现派生从QObject的子类中的内省方法。

  1. // defined at ..\Qt\src\corelib\kernel\qobjectdefs.h   
  2. /* tmake ignore Q_OBJECT */   
  3. #define Q_OBJECT \   
  4. public: \     
  5.     Q_OBJECT_CHECK \     
  6. static const QMetaObject staticMetaObject; \     
  7.     Q_OBJECT_GETSTATICMETAOBJECT \     
  8. virtual const QMetaObject *metaObject() const; \     
  9. virtual void *qt_metacast(const char *); \     
  10.     QT_TR_FUNCTIONS \     
  11. virtual int qt_metacall(QMetaObject::Call, int, void **); \  

此外,所有的Qt widgets类均继承自QObject, QObject所提供的isWidgetType自省方法可以很方便让QObject子对象查询自己是否是wideget, 而且它会比 qobject_cast(obj) 或者 obj->inherits快很多。原因qobject_cast()t和inherits()都是借助元对象系统来实现其功能的,isWidgetType()是QObject本身的标志位得以实现。

更多自省方法定义在QMetaObject,以下是QMetaObject声明的源代码:

  1. struct Q_CORE_EXPORT QMetaObject     
  2. {     
  3. const char *className() const;     
  4. const QMetaObject *superClass() const;     
  5.     QObject *cast(QObject *obj) const;     
  6.  
  7.     ....     
  8. int methodOffset() const;     
  9. int enumeratorOffset() const;     
  10. int propertyOffset() const;     
  11. int classInfoOffset() const;     
  12. int constructorCount() const;     
  13. int methodCount() const;     
  14. int enumeratorCount() const;     
  15. int propertyCount() const;     
  16. int classInfoCount() const;     
  17. int indexOfConstructor(const char *constructor) const;     
  18. int indexOfMethod(const char *method) const;     
  19. int indexOfSignal(const char *signal) const;     
  20. int indexOfSlot(const char *slot) const;     
  21. int indexOfEnumerator(const char *name) const;     
  22. int indexOfProperty(const char *name) const;     
  23. int indexOfClassInfo(const char *name) const;     
  24.     ...     
  25. }   

上述方法主要是实现对元对象表的访问及其操作,对元对象表(由moc实现)实例如下所示:

  1. // defined at ..\Qt\src\corelib\kernel\qobjectdefs.h   
  2. /* tmake ignore Q_OBJECT */   
  3. #define Q_OBJECT \   
  4. public: \    
  5.     Q_OBJECT_CHECK \    
  6. static const QMetaObject staticMetaObject; \    
  7.     Q_OBJECT_GETSTATICMETAOBJECT \    
  8. virtual const QMetaObject *metaObject() const; \    
  9. virtual void *qt_metacast(const char *); \    
  10.     QT_TR_FUNCTIONS \    
  11. virtual int qt_metacall(QMetaObject::Call, int, void **); \   

总结:

1、Qt是通过QObject、QMetaObject类实现其内省机制,

2、QObject暴露给用户的共有自省方法有objectName(), inherits(), isWidgetType()等

3、大多数自省方法是QObject派发给QMetaObject实现 (e.g. QMetaObject::className,),元对象模型编译器moc负责自省方法的实现

4、更多自省方法定义在QMetaObject,而是为了等信号槽通讯、事件派发等机制,

小结:关于解析 Qt 内省机制剖析的内容介绍完了,希望本文对你有所帮助!

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
Windows恶意软件20年“... 在Windows的早期年代,病毒游走于系统之间,偶尔删除文件(但被删除的文件几乎都是可恢复的),并弹...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
着眼MAC地址,解救无法享受D... 在安装了DHCP服务器的局域网环境中,每一台工作站在上网之前,都要先从DHCP服务器那里享受到地址动...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...