提供了一堆原来C++没有的功能,比如反射什么的。。。但是可能还是没有Delphi的类之类更强,因为类之类可以“创建类”。可惜我学艺不精,对“类之类”也没有完全学会。先留个爪,有空把两个东西都好好学学,再来个比较。。。

-----------------------------------------------------------------------------------

除了D指针,Qt中另一个很有意思的部分就是Q_OBJECT宏了。该宏提供了对元对象的访问,使得能够使用比如信号和槽等QObject的更多特性。元对象提供了诸如类名、属性和方法等的信息,也被称为“反射”。

通过使用QMetaObject,我们能够用如下代码显示一些类的信息:

  1. QObject obj;
  2. const QMetaObject *metaObj = obj.metaObject();
  3. qDebug() << "class name: " << metaObj->className();
  4. qDebug() << "class info count: " << metaObj->classInfoCount();
  5. qDebug() << "methods: ";
  6. // 从QMetaObject::methodOffset()开始打印,使其不会显示父类的方法
  7. for (int i = metaObj->methodOffset(); i < metaObj->methodCount(); ++i)
  8. qDebug() << metaObj->method(i).methodType() << " " << metaObj->method(i).signature();

由于C++并没有提供对这些信息的任何支持,Qt引入了元对象编译器(moc)来完成相应的工作。moc会读取每个头文件,如果发现其中定义的类是继承自QObject,且定义了Q_OBJECT宏,便会创建一个相应的C++源代码文件(moc_*.cpp),来完成这些工作。通过代码生成的工作,Qt不仅能够获得诸如Java等语言的灵活性,还能很好的保证继承自C++的性能和可扩展性

假设我们有如下所示的简单类:

  1. class MyObject : public QObject
  2. {
  3. Q_OBJECT
  4. public:
  5. explicit MyObject(QObject *parent = 0);
  6. void myFunc();
  7. public slots:
  8. void mySlot(int myParam);
  9. signals:
  10. void mySignal(int myParam);
  11. };

moc会自动创建以下信息:

  1. // 保存在QMetaObject::d.data指向的空间,其起始部分是一个QMetaObjectPrivate结构体
  2. static const uint qt_meta_data_MyObject[] = {
  3. 5,       // 版本号,其内部结构在Qt开发中有所改变
  4. 0,       // 类名,其值为字符串qt_meta_stringdata_MyObject的偏移量
  5. // 以下值为(数量,索引)对
  6. 0,    0, // 类信息
  7. 2,   14, // 这里定义了两个方法,其起始索引为14(即signal部分)
  8. 0,    0, // 属性
  9. 0,    0, // 枚举
  10. 0,    0, // 构造函数
  11. 0,       // 标识
  12. 1,       // signal数量
  13. // 对于signal、slot和property,其signature和parameters为字符串qt_meta_stringdata_MyObject的偏移量
  14. // signals: signature, parameters, type, tag, flags
  15. 18,   10,    9,    9, 0x05,
  16. // slots: signature, parameters, type, tag, flags
  17. 32,   10,    9,    9, 0x0a,
  18. 0        // eod
  19. };
  20. // 保存在QMetaObject::d.stringdata指向的空间
  21. static const char qt_meta_stringdata_MyObject[] = {
  22. "MyObject/0/0myParam/0mySignal(int)/0"
  23. "mySlot(int)/0"
  24. };

以上信息,及其基类的相关信息,都保存在该类对应的元对象中:

  1. const QMetaObject MyObject::staticMetaObject = {
  2. { &QObject::staticMetaObject, // 指向其基类的元对象,保存在QMetaObject::d.superdata
  3. qt_meta_stringdata_MyObject, qt_meta_data_MyObject, 0 }
  4. };

这样,如果我们希望对QObject的对象进行类型转换,就不需使用开销较大的运算符dynamic_cast, 而能够直接使用qobject_cast。该模板函数利用了元对象系统的信息,避免了在运行时进行类型转换:

  1. template <class T> inline T qobject_cast(QObject *object)
  2. {
  3. #if !defined(QT_NO_QOBJECT_CHECK)
  4. reinterpret_cast(0)->qt_check_for_QOBJECT_macro(*reinterpret_cast(object));
  5. #endif
  6. return static_cast(reinterpret_cast(0)->staticMetaObject.cast(object));
  7. }

这里,目标类型的元对象仅仅检查其是否从自身继承而来:

  1. const QObject *QMetaObject::cast(const QObject *obj) const
  2. {
  3. if (obj) {
  4. const QMetaObject *m = obj->metaObject();
  5. do {
  6. if (m == this)
  7. return obj;
  8. } while ((m = m->d.superdata));
  9. }
  10. return 0;
  11. }

此外,moc会为每一个信号创建相应函数。当信号被emit时,该函数会被自动调用:

  1. void MyObject::mySignal(int _t1)
  2. {
  3. void *_a[] = { 0, const_cast<void*>(reinterpret_cast<const void*>(&_t1)) };
  4. // 检查链接到该信号的所有slot,并根据链接类型进行调用
  5. QMetaObject::activate(this, &staticMetaObject, 0, _a);
  6. }

最后,这些信号都会通过moc创建的qt_metacall函数被调用:

  1. int MyObject::qt_metacall(QMetaObject::Call _c, int _id, void **_a)
  2. {
  3. // 如果该函数已被基类调用,则直接返回
  4. _id = QObject::qt_metacall(_c, _id, _a);
  5. if (_id < 0)
  6. return _id;
  7. // 根据函数的ID进行调用
  8. if (_c == QMetaObject::InvokeMetaMethod) {
  9. switch (_id) {
  10. case 0: mySignal((*reinterpret_cast< int(*)>(_a[1]))); break;
  11. case 1: mySlot((*reinterpret_cast< int(*)>(_a[1]))); break;
  12. default: ;
  13. }
  14. // 删除被该类“消耗”的ID,使得其子类类在处理时ID总是从0开始,而返回值-1则表示该函数已被调用
  15. _id -= 2;
  16. }
  17. return _id;
  18. }

参考:http://blog.csdn.net/seanyxie/article/details/6120040

QMetaObject感觉跟Delphi的类之类有一拼,好好学一下的更多相关文章

  1. delphi 实体类 JSON 数组

    delphi 实体类 与JSON转换,序列化 TJson REST.JSON.pas   TJson.JsonToObjectTJson.ObjectToJsonString JsonEncode O ...

  2. 比较C++、Java、Delphi声明类对象时候的相关语法

    同学们在学习的时候经常会遇到一些问题,C++.Java.Delphi他们到底有什么不一样的呢?今天我们来比较C++.Java.Delphi声明类对象时候的相关语法.希望对大家有帮助! C++中创建对象 ...

  3. Delphi 遍历类中的属性

    http://blog.csdn.net/easyboot/article/details/8004954 Delphi 遍历类中的属性 标签: delphistringbuttonclassform ...

  4. 转:Delphi的类与继承(VB与delphi比较)

    既然已经做出了com程序用delphi来开发的决定,那当然就要对delphi进行一些深入的了解.有人说delphi是一个用控件堆砌起来的工具,和vb没什么两样:也有人说dephi实际上是面向过程的,他 ...

  5. 再次深入理解delphi的类

    property WindowState: TWindowState read FWindowState write SetWindowState; {声明一个属性WindowState,它从字段FW ...

  6. 获取Delphi所有类的类信息

    Delphi遍历进程中所有Class的TypeInfo,即便是在implementation中的class或者其他 class的private的子class. 一般普通EXE中的TypeInfo存放在 ...

  7. DELPHI学习---类和对象(五篇)

    Classes and objects(类和对象) 类(或者类类型)定义了一个结构,它包括字段(也称为域).方法和属性:类的实例叫做对象:类的字段.方法和属性被称为它的部件(components)或成 ...

  8. 理解 Delphi 的类(十一) - 深入类中的方法[8] - 抽象方法与抽象类

    //抽象方法类似与接口; 在没有接口的年代 Delphi 是用抽象方法来模拟接口的; 我想它最终会被接口替代. {下面就定义了两个抽象方法} TMyClass = class(TObject)   p ...

  9. delphi TComponent类 2

    来自:http://blog.csdn.net/lailai186/article/details/7442385 ------------------------------------------ ...

随机推荐

  1. 如何在Byte[]和String之间进行转换

    源自C#与.NET程序员面试宝典. 如何在Byte[]和String之间进行转换? 比特(b):比特只有0 1,1代表有脉冲,0代表无脉冲.它是计算机物理内存保存的最基本单元. 字节(B):8个比特, ...

  2. Google Map API v2 (三)----- 地图上添加标记(Marker),标记info窗口,即指定经纬度获取地址字符串

    接上篇 http://www.cnblogs.com/inkheart0124/p/3536322.html 1,在地图上打个标记 private MarkerOptions mMarkOption; ...

  3. php yield

    php中关于 yield 关键字的介绍[点击查看] <?php function gen_one_to_three() { for ($i = 1; $i <= 3; $i++) { // ...

  4. Pagekit安装

    Pagekit 是一个模块化,轻量的 CMS 系统,基于现代化的技术,如 Symfony 组件和 Doctrine.它提供了一个很好的平台,用于主题和延伸开发.Pagekit 为您提供了工具来创造美丽 ...

  5. HTML+CSS基础学习笔记(1)

    一.了解HTML.CSS.JS 1.HTML是网页内容的载体. 内容就是网页制作者放在页面上想要让用户浏览的信息,可以包含文字.图片.视频等. 2.CSS样式是表现. 用来改变内容外观的东西称之为表现 ...

  6. C++话题

    1.多态地实现 A:C++中多态的实现原理是怎样的? Q:通过迟邦定技术(late binding)实现. 具体实现原理如下: 1. 基类中函数带virtual关键字,表示该方法为虚函数. 2. 子类 ...

  7. JS事件监听 JS:attachEvent和addEventListener 使用方法

    attachEvent与addEventListener区别适应的浏览器版本不同,同时在使用的过程中要注意attachEvent方法          按钮onclickaddEventListene ...

  8. 在CentOS6.0上安装Oracle 11gR2 (11.2.0.1)以及基本的配置(一)

    首先安装CentOS6.0   就不用说了.安装即可.唯一需要注意的就是后面Oracle 11G Installation guide中的Checking the Software Requireme ...

  9. 如何改app图标名称

    InfoPlist.strings文件里写上:       CFBundleDisplayName="中文名字";

  10. 3 委托、匿名函数、lambda表达式

    委托.匿名函数.lambda表达式 在 2.0 之前的 C# 版本中,声明委托的唯一方法是使用命名方法.C# 2.0 引入了匿名方法,而在 C# 3.0 及更高版本中,Lambda 表达式取代了匿名方 ...