Q_OBJECT宏的作用
        Qt 中,如果一个类中定义了信号(signals)或槽(slots),那么这个类必须包含 Q_OBJECT 宏。
Q_OBJECT宏是 Qt 元对象系统的核心部分,它使得信号和槽机制能够正常工作。
Q_OBJECT宏是 Qt 的元对象系统的一部分,它会生成额外的代码,用于支持信号和槽的功能,包括信号的注册、连接、断开以及对象的动态属性等。
如果类中定义了信号或槽,但没有包含Q_OBJECT宏,编译器会报错。
Q_OBJECT 宏的作用主要体现在以下几个方面:
1. 元对象的生成
当一个类继承自 QObject 并包含 Q_OBJECT 宏时,Qt 的元对象系统会为这个类生成一个元对象(QMetaObject)。元对象包含了类的以下信息:
-  类名:类的名称。 
-  信号列表:类中定义的所有信号。 
-  槽列表:类中定义的所有槽。 
-  属性列表:类中定义的所有属性。 
-  父类信息:类的继承关系。 
        元对象的生成是通过 Qt 的 元对象编译器(moc) 实现的。moc 是一个预处理器,它会解析包含 Q_OBJECT 宏的头文件,并生成一个额外的 C++ 文件(通常是 .moc 文件)。
这个生成的文件包含了元对象的实现代码。
2. 信号和槽
Q_OBJECT 宏使得信号和槽机制能够正常工作。具体来说:
-  信号的定义和发出: -  信号是类的成员函数,但它们的行为与普通函数不同。信号的定义需要在类中使用 signals:关键字。
-  当信号被发出(使用 emit关键字)时,Qt 的元对象系统会根据信号的名称和参数类型查找连接的槽,并调用这些槽。
-  例如: 
 
-  
signals:void mySignal(int value);-  -  调用 emit mySignal(42);时,Qt 会查找所有连接到mySignal的槽,并将参数42传递给这些槽。
 
-  
-  槽的定义和连接: -  槽是类的普通成员函数,但它们可以通过 QObject::connect方法与信号连接。
-  槽的定义需要在类中使用 slots:关键字(或直接使用普通成员函数)。
-  例如: 
 
-  
public slots:void mySlot(int value);当信号被发出时,连接到该信号的槽会被调用:
connect(sender, &Sender::mySignal, receiver, &Receiver::mySlot);3. 运行时类型信息(RTTI)
    Q_OBject宏为类提供了运行时类型信息(RTTI)。
通过元对象系统,Qt 可以在运行时查询类的名称、继承关系、信号和槽的信息。例如:
-  获取类名: 
QString className = obj->metaObject()->className();检查继承关系:
if (obj->inherits("MyClass")) {// obj 是 MyClass 或其子类的实例
}4. 动态属性
     Q_OBJECT 宏使得类的实例可以动态地添加和访问属性。这些属性可以在运行时设置和查询,而不需要在类中显式定义。例如:
obj->setProperty("myProperty", 42);
int value = obj->property("myProperty").toInt();5. 元对象的实现
Q_OBJECT 宏的实现主要依赖于以下几个部分:
-  moc(元对象编译器): -  moc 是一个预处理器,它会解析包含 Q_OBJECT宏的头文件,并生成一个额外的 C++ 文件。
-  生成的文件包含了元对象的实现代码,包括信号和槽的注册、事件处理等。 
 
-  
-  QMetaObject类:-  QMetaObject是元对象系统的中心类,它封装了类的元信息。
-  每个包含 Q_OBJECT宏的类都会有一个静态的QMetaObject实例,可以通过metaObject()方法访问。
 
-  
-  QObject的构造函数:-  在 QObject的构造函数中,会调用元对象系统初始化代码,将对象实例与元对象关联起来。
 
-  
