Qt Core - The Meta-Object System
Qt的元对象系统为对象间通信提供了信号和槽机制、运行时类型信息和动态属性系统。
元对象系统基于三个方面:
- QObject 为对象提供了一个可以利用 Meta-Object 系统基类。
- 在类声明的私有部分(private)中使用 Q_OBJECT 宏的方法, 被用于启用 Meta-Object 功能,如动态属性、信号和槽。
- 元对象编译器(moc)为每个 QObject 子类提供实现 Meta-Object 功能所需的代码。
MOC工具读取一个C++源文件。如果它在类的定义中找到 Q_OBJECT 宏,则会生成另一个C++源文件,其中包含了 meta-object 的功能代码。生成的源文件要么包含在类的源文件中,要么更常见地编译并链接到类的实现代码中。
理解: Qt 代码中如果包含 Q_OBJECT 宏, moc(一个集成的编译工具) 会在 C++ 编译器工作之前进行"翻译", 也就是重新生成一个与Qt源文件对应的C++源文件, 这个源文件使用通用C++代码实现了 Meta-Object 功能. 最后, C++ 编译器编译链接的时使用的是翻译后的C++代码文件.
除了提供对象间通信的信号和插槽机制(引入系统的主要原因),元对象代码还提供以下附加功能:
- QObject::metaObject() 返回类的关联元对象。
- QMetaObject::className() 在运行时将类名返回为字符串,而不需要通过C++编译器支持本机运行时类型信息(RTTI)。
- QObject::inherits() 返回对象是否是继承QObject继承树中指定类的类的实例。
- QObject::tr() and QObject::trUtf8() 将字符串转换为国际化格式。
- QObject::setProperty() and QObject::property() 名称动态设置和获取属性。
- QMetaObject::newInstance() 构造类的新实例。
也可以对QObject类使用 qobject_cast() 执行动态强制转换。qobject_cast() 函数的行为类似于标准C++的dynamic_cast(),其优点是不需要RTTI支持,并且它可以跨动态库边界工作。它尝试将其参数强制转换为尖括号中指定的指针类型,如果对象的类型正确(在运行时确定),则返回非零指针;如果对象的类型不兼容,则返回0。
虽然可以使用 QObject 作为没有 Q_OBJECT 宏和元对象代码的基类,但如果不使用 Q_OBJECT 宏,则此处描述的信号和槽以及其他功能都将不可用。从元对象系统的角度来看,没有元代码的QObject子类等价于它与元对象代码最接近的祖先。例如,这意味着 QMetaObject::ClassName() 不会返回类的实际名称,而是返回该祖先的类名。