博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
读QT5.7源码(三)Q_OBJECT 和QMetaObject
阅读量:6413 次
发布时间:2019-06-23

本文共 18011 字,大约阅读时间需要 60 分钟。

Qt meta-object系统基于三个方面:

  1、QObject提供一个基类,方便派生类使用meta-object系统的功能;
  2、Q_OBJECT宏,在类的声明体内激活meta-object功能,比如动态属性、信号、槽;
  3、Meta Object编译器(MOC),为每个QObject派生类生成代码,以支持meta-object功能。

QObject定义了从一个QObject对象访问meta-object功能的接口,Q_OBJECT宏用来告诉编译器该类需要激活meta- object功能,编译器在扫描一个源文件时,如果发现类的声明中有这个宏,就会生成一些代码来为支持meta-object功能——主要是生成该类对应 MetaObject类以及对QObject的函数override。

QObject和QMetaObject:

顾名思义,QMetaObject包含了QObject的所谓的元数据,也就是QObject信息的一些描述信息:除了类型信息外,还包含QT中特 有的signal&slot信息。

 QObject::metaObject ()方法返回一个QObject对象对应的metaobject对象,注意这个方法是virtual方法。如上文所说,如果一个类的声明中包含了 Q_OBJECT宏,编译器会生成代码来实现这个类对应的QMetaObject类,并重载QObject::metaObject()方法来返回这个 QMetaObject类的实例引用。这样当通过QObject类型的引用调用metaObejct方法时,返回的是这个引用的所指的真实对象的 metaobject。

如果一个类从QObject派生,确没有声明Q_OBJECT宏,那么这个类的metaobject对象不会被生成,这样这个类所声明的 signal slot都不能使用,而这个类实例调用metaObject()返回的就是其父类的metaobject对象,这样导致的后果就是你从这个类实例获得的元 数据其实都是父类的数据,这显然给你的代码埋下隐患。因此如果一个类从QOBject派生,它都应该声明Q_OBJECT宏,不管这个类有没有定义 signal&slot和Property。

这样每个QObject类都有一个对应的QMetaObject类,形成一个平行的类型层次。

QMetaObject提供的信息:

下面通过QMetaObject的接口来解读QMetaObject提供的信息:

1、基本信息
     const char * className () const;
     const QMetaObject * superClass () const
2、classinfo: 提供额外的类信息。其实就是一些名值对。 用户可以在类的声明中以Q_CLASSINFO(name, value)方式添加。
      int classInfoCount () const
      int classInfoOffset () const
      QMetaClassInfo classInfo ( int index ) const
      int indexOfClassInfo ( const char * name ) const
3、contructor:提供该类的构造方法信息
     QMetaMethod constructor ( int index ) const
     int constructorCount () const
     int indexOfConstructor ( const char * constructor ) const
4、enum:描述该类声明体中所包含的枚举类型信息
    QMetaEnum enumerator ( int index ) const
    int enumeratorCount () const
    int enumeratorOffset () const
    int indexOfEnumerator ( const char * name ) const
5、method:描述类中所包含方法信息:包括property,signal,slot等,包括祖先类,如何组织暂时不确定。
    QMetaMethod method ( int index ) const
    int methodCount () const
    int methodOffset () const
    int indexOfMethod ( const char * method ) const
    int indexOfSignal ( const char * signal ) const
    int indexOfSlot ( const char * slot ) const
6、property:类型的属性信息
     QMetaProperty property ( int index ) const
     int propertyCount () const
     int propertyOffset () const
     int indexOfProperty ( const char * name ) const
     QMetaProperty userProperty () const  //返回类中设置了USER flag的属性,(难道只能有一个这样的属性?)

注意:对于类里面定义的函数,构造函数,枚举,只有加上一些宏才表示你希望为方法提供meta信息。比如 Q_ENUMS用来注册宏,

Q_INVACABLE用来注册方法(包括构造函数)。Qt这么设计的原因应该是避免meta信息的臃肿。

 

下面是Q_OBJECT的宏定义

#define Q_OBJECT \public: \    Q_OBJECT_CHECK \    QT_WARNING_PUSH \    Q_OBJECT_NO_OVERRIDE_WARNING \    static const QMetaObject staticMetaObject; \    virtual const QMetaObject *metaObject() const; \    virtual void *qt_metacast(const char *); \    virtual int qt_metacall(QMetaObject::Call, int, void **); \    QT_TR_FUNCTIONS \private: \    Q_OBJECT_NO_ATTRIBUTES_WARNING \    Q_DECL_HIDDEN_STATIC_METACALL static void qt_static_metacall(QObject *, QMetaObject::Call, int, void **); \    QT_WARNING_POP \    struct QPrivateSignal {}; \    QT_ANNOTATE_CLASS(qt_qobject, "")

 

因此:

每一个定义了Q_OBJECT宏的,直接或者间接继承于QObject的类 有包含一个名为staticMetaObject的静态QMetaObject对象, 一个私有的静态函数qt_static_metacall。

它们为该类型的所有对象共有,同时它也继承了每一个祖父对象这两个静态成员和函数。

    virtual const QMetaObject *metaObject() const; \    用于获取类静态拥有的元对象

    virtual void *qt_metacast(const char *); \                   通过元对象获取对象指针
    virtual int qt_metacall(QMetaObject::Call, int, void **); \     用于信号槽机制

每个不同的类 在MOC生成代码的时候会重写写这些虚函数。

QMetaObjectl类的定义

struct Q_CORE_EXPORT QMetaObject{    class Connection;    const char *className() const;    const QMetaObject *superClass() const;     bool inherits(const QMetaObject *metaObject) const Q_DECL_NOEXCEPT;    QObject *cast(QObject *obj) const;    const QObject *cast(const QObject *obj) const; #ifndef QT_NO_TRANSLATION    QString tr(const char *s, const char *c, int n = -1) const;#endif // QT_NO_TRANSLATION     int methodOffset() const;    int enumeratorOffset() const;    int propertyOffset() const;    int classInfoOffset() const;     int constructorCount() const;    int methodCount() const;    int enumeratorCount() const;    int propertyCount() const;    int classInfoCount() const;     int indexOfConstructor(const char *constructor) const;    int indexOfMethod(const char *method) const;    int indexOfSignal(const char *signal) const;    int indexOfSlot(const char *slot) const;    int indexOfEnumerator(const char *name) const;    int indexOfProperty(const char *name) const;    int indexOfClassInfo(const char *name) const;     QMetaMethod constructor(int index) const;    QMetaMethod method(int index) const;    QMetaEnum enumerator(int index) const;    QMetaProperty property(int index) const;    QMetaClassInfo classInfo(int index) const;    QMetaProperty userProperty() const;     static bool checkConnectArgs(const char *signal, const char *method);    static bool checkConnectArgs(const QMetaMethod &signal,                                 const QMetaMethod &method);    static QByteArray normalizedSignature(const char *method);    static QByteArray normalizedType(const char *type);     // internal index-based connect    static Connection connect(const QObject *sender, int signal_index,                        const QObject *receiver, int method_index,                        int type = 0, int *types = Q_NULLPTR);    // internal index-based disconnect    static bool disconnect(const QObject *sender, int signal_index,                           const QObject *receiver, int method_index);    static bool disconnectOne(const QObject *sender, int signal_index,                              const QObject *receiver, int method_index);    // internal slot-name based connect    static void connectSlotsByName(QObject *o);     // internal index-based signal activation    static void activate(QObject *sender, int signal_index, void **argv);    static void activate(QObject *sender, const QMetaObject *, int local_signal_index, void **argv);    static void activate(QObject *sender, int signal_offset, int local_signal_index, void **argv);     static bool invokeMethod(QObject *obj, const char *member,                             Qt::ConnectionType,                             QGenericReturnArgument ret,                             QGenericArgument val0 = QGenericArgument(Q_NULLPTR),                             QGenericArgument val1 = QGenericArgument(),                             QGenericArgument val2 = QGenericArgument(),                             QGenericArgument val3 = QGenericArgument(),                             QGenericArgument val4 = QGenericArgument(),                             QGenericArgument val5 = QGenericArgument(),                             QGenericArgument val6 = QGenericArgument(),                             QGenericArgument val7 = QGenericArgument(),                             QGenericArgument val8 = QGenericArgument(),                             QGenericArgument val9 = QGenericArgument());     static inline bool invokeMethod(QObject *obj, const char *member,                             QGenericReturnArgument ret,                             QGenericArgument val0 = QGenericArgument(Q_NULLPTR),                             QGenericArgument val1 = QGenericArgument(),                             QGenericArgument val2 = QGenericArgument(),                             QGenericArgument val3 = QGenericArgument(),                             QGenericArgument val4 = QGenericArgument(),                             QGenericArgument val5 = QGenericArgument(),                             QGenericArgument val6 = QGenericArgument(),                             QGenericArgument val7 = QGenericArgument(),                             QGenericArgument val8 = QGenericArgument(),                             QGenericArgument val9 = QGenericArgument())    {        return invokeMethod(obj, member, Qt::AutoConnection, ret, val0, val1, val2, val3,                val4, val5, val6, val7, val8, val9);    }     static inline bool invokeMethod(QObject *obj, const char *member,                             Qt::ConnectionType type,                             QGenericArgument val0 = QGenericArgument(Q_NULLPTR),                             QGenericArgument val1 = QGenericArgument(),                             QGenericArgument val2 = QGenericArgument(),                             QGenericArgument val3 = QGenericArgument(),                             QGenericArgument val4 = QGenericArgument(),                             QGenericArgument val5 = QGenericArgument(),                             QGenericArgument val6 = QGenericArgument(),                             QGenericArgument val7 = QGenericArgument(),                             QGenericArgument val8 = QGenericArgument(),                             QGenericArgument val9 = QGenericArgument())    {        return invokeMethod(obj, member, type, QGenericReturnArgument(), val0, val1, val2,                                 val3, val4, val5, val6, val7, val8, val9);    }     static inline bool invokeMethod(QObject *obj, const char *member,                             QGenericArgument val0 = QGenericArgument(Q_NULLPTR),                             QGenericArgument val1 = QGenericArgument(),                             QGenericArgument val2 = QGenericArgument(),                             QGenericArgument val3 = QGenericArgument(),                             QGenericArgument val4 = QGenericArgument(),                             QGenericArgument val5 = QGenericArgument(),                             QGenericArgument val6 = QGenericArgument(),                             QGenericArgument val7 = QGenericArgument(),                             QGenericArgument val8 = QGenericArgument(),                             QGenericArgument val9 = QGenericArgument())    {        return invokeMethod(obj, member, Qt::AutoConnection, QGenericReturnArgument(), val0,                val1, val2, val3, val4, val5, val6, val7, val8, val9);    }     QObject *newInstance(QGenericArgument val0 = QGenericArgument(Q_NULLPTR),                         QGenericArgument val1 = QGenericArgument(),                         QGenericArgument val2 = QGenericArgument(),                         QGenericArgument val3 = QGenericArgument(),                         QGenericArgument val4 = QGenericArgument(),                         QGenericArgument val5 = QGenericArgument(),                         QGenericArgument val6 = QGenericArgument(),                         QGenericArgument val7 = QGenericArgument(),                         QGenericArgument val8 = QGenericArgument(),                         QGenericArgument val9 = QGenericArgument()) const;     enum Call {        InvokeMetaMethod,        ReadProperty,        WriteProperty,        ResetProperty,        QueryPropertyDesignable,        QueryPropertyScriptable,        QueryPropertyStored,        QueryPropertyEditable,        QueryPropertyUser,        CreateInstance,        IndexOfMethod,        RegisterPropertyMetaType,        RegisterMethodArgumentMetaType    };     int static_metacall(Call, int, void **) const;    static int metacall(QObject *, Call, int, void **);     struct { // private data        const QMetaObject *superdata;        const QByteArrayData *stringdata;        const uint *data;        typedef void (*StaticMetacallFunction)(QObject *, QMetaObject::Call, int, void **);        StaticMetacallFunction static_metacall;        const QMetaObject * const *relatedMetaObjects;        void *extradata; //reserved for future use    } d;};

 

struct { // private data        const QMetaObject *superdata;        const QByteArrayData *stringdata;        const uint *data;        typedef void (*StaticMetacallFunction)(QObject *, QMetaObject::Call, int, void **);        StaticMetacallFunction static_metacall;        const QMetaObject * const *relatedMetaObjects;        void *extradata; //reserved for future use    } d;

 

元对象 的所有数据都由该结构定义,

 

1、const QMetaObject *superdata;    父类的staticMetaObject指针

2、QByteArrayData *stringdata;         字符串数组,保存类的 类名,槽函数名  信号函数名等 字符串信息。

3、const uint *data;                               无符号整形数组,该数组是个预定义的复合数据结构,由QMetaObjectPrivate 类提供管理,保存了类的基本信息,和一些索引值

4、const QMetaObject * const *relatedMetaObjects;  和void *extradata;    MOC不会为他们生成对应的数据

MOC 为一个类生成元数据例子。类widget 包含三个信号 和 槽,分别为 

protected slots:

    void TestSlot(QString &str);
signals:
    void TestSignal1(QString &str);
    void TestSignal2(QString &str,int i);
    void TestSignal3();

MOC生成的代码

QT_BEGIN_MOC_NAMESPACEstruct qt_meta_stringdata_Widget_t {    QByteArrayData data[9];    char stringdata0[68];};#define QT_MOC_LITERAL(idx, ofs, len) \    Q_STATIC_BYTE_ARRAY_DATA_HEADER_INITIALIZER_WITH_OFFSET(len, \    qptrdiff(offsetof(qt_meta_stringdata_Widget_t, stringdata0) + ofs \        - idx * sizeof(QByteArrayData)) \    )static const qt_meta_stringdata_Widget_t qt_meta_stringdata_Widget = {    {                                                             //对应QMetaObject的stringdata  ,QT_MOC_LITERAL宏来构成QByteArray对象,每个QByteArray对QT_MOC_LITERAL(0, 0, 6), // "Widget"                                应一个下面\0结尾的字符串,以此方便对字符串数据的管理QT_MOC_LITERAL(1, 7, 11), // "TestSignal1"                         QT_MOC_LITERAL(2, 19, 0), // ""QT_MOC_LITERAL(3, 20, 8), // "QString&"QT_MOC_LITERAL(4, 29, 3), // "str"QT_MOC_LITERAL(5, 33, 11), // "TestSignal2"QT_MOC_LITERAL(6, 45, 1), // "i"QT_MOC_LITERAL(7, 47, 11), // "TestSignal3"QT_MOC_LITERAL(8, 59, 8) // "TestSlot"     },    "Widget\0TestSignal1\0\0QString&\0str\0"    "TestSignal2\0i\0TestSignal3\0TestSlot"};#undef QT_MOC_LITERAL static const uint qt_meta_data_Widget[] = {

   说明:QT_MOC_LITERAL(0, 0, 6), // "Widget"  (0:序号,0,:字符("Widget\0T...")在字符串中的起始位置,6:字符长度)。

 

// content:       7,       // revision       0,       // classname       0,    0, // classinfo       4,   14, // methods       0,    0, // properties       0,    0, // enums/sets       0,    0, // constructors       0,       // flags       3,       // signalCount  // signals: name, argc, parameters, tag, flags     //其中第一值是QByteArrayData数组的索引值,以此来找到对应的字符串名称,第二个值是参数个数,第三个是参数       1,    1,   34,    2, 0x06 /* Public */,     //的大小       5,    2,   37,    2, 0x06 /* Public */,       7,    0,   42,    2, 0x06 /* Public */,  // slots: name, argc, parameters, tag, flags       8,    1,   43,    2, 0x09 /* Protected */,  // signals: parameters    QMetaType::Void, 0x80000000 | 3,    4,    QMetaType::Void, 0x80000000 | 3, QMetaType::Int,    4,    6,    QMetaType::Void,  // slots: parameters    QMetaType::Void, 0x80000000 | 3,    4,        0        // eod}; void Widget::qt_static_metacall(QObject *_o, QMetaObject::Call _c, int _id, void **_a)    //以名称或索引方式调用对应的信号函数{    if (_c == QMetaObject::InvokeMetaMethod) {        Widget *_t = static_cast
(_o); Q_UNUSED(_t) switch (_id) { case 0: _t->TestSignal1((*reinterpret_cast< QString(*)>(_a[1]))); break; case 1: _t->TestSignal2((*reinterpret_cast< QString(*)>(_a[1])),(*reinterpret_cast< int(*)>(_a[2]))); break; case 2: _t->TestSignal3(); break; case 3: _t->TestSlot((*reinterpret_cast< QString(*)>(_a[1]))); break; default: ; } } else if (_c == QMetaObject::IndexOfMethod) { int *result = reinterpret_cast
(_a[0]); void **func = reinterpret_cast
(_a[1]); { typedef void (Widget::*_t)(QString & ); if (*reinterpret_cast<_t *>(func) == static_cast<_t>(&Widget::TestSignal1)) { *result = 0; return; } } { typedef void (Widget::*_t)(QString & , int ); if (*reinterpret_cast<_t *>(func) == static_cast<_t>(&Widget::TestSignal2)) { *result = 1; return; } } { typedef void (Widget::*_t)(); if (*reinterpret_cast<_t *>(func) == static_cast<_t>(&Widget::TestSignal3)) { *result = 2; return; } } }} const QMetaObject Widget::staticMetaObject = { { &QWidget::staticMetaObject, qt_meta_stringdata_Widget.data, qt_meta_data_Widget, qt_static_metacall, Q_NULLPTR, Q_NULLPTR} //初始化静态对象staticMetaObject}; const QMetaObject *Widget::metaObject() const{ return QObject::d_ptr->metaObject ? QObject::d_ptr->dynamicMetaObject() : &staticMetaObject;} void *Widget::qt_metacast(const char *_clname){ if (!_clname) return Q_NULLPTR; if (!strcmp(_clname, qt_meta_stringdata_Widget.stringdata0)) return static_cast
(const_cast< Widget*>(this)); return QWidget::qt_metacast(_clname);} int Widget::qt_metacall(QMetaObject::Call _c, int _id, void **_a){ _id = QWidget::qt_metacall(_c, _id, _a); if (_id < 0) return _id; if (_c == QMetaObject::InvokeMetaMethod) { if (_id < 4) qt_static_metacall(this, _c, _id, _a); _id -= 4; } else if (_c == QMetaObject::RegisterMethodArgumentMetaType) { if (_id < 4) *reinterpret_cast
(_a[0]) = -1; _id -= 4; } return _id;} // SIGNAL 0void Widget::TestSignal1(QString & _t1){ void *_a[] = { Q_NULLPTR, const_cast
(reinterpret_cast
(&_t1)) }; QMetaObject::activate(this, &staticMetaObject, 0, _a);} // SIGNAL 1void Widget::TestSignal2(QString & _t1, int _t2){ void *_a[] = { Q_NULLPTR, const_cast
(reinterpret_cast
(&_t1)), const_cast
(reinterpret_cast
(&_t2)) }; QMetaObject::activate(this, &staticMetaObject, 1, _a);} // SIGNAL 2void Widget::TestSignal3(){ QMetaObject::activate(this, &staticMetaObject, 2, Q_NULLPTR);}QT_END_MOC_NAMESPACE

 

其中  static const uint qt_meta_data_Widget[] = {

 // content:
       7,       // revision
       0,       // classname
       0,    0, // classinfo
       4,   14, // methods
       0,    0, // properties
       0,    0, // enums/sets
       0,    0, // constructors
       0,       // flags
       3,       // signalCount

 

 

对应

 

struct QMetaObjectPrivate

{
    enum { OutputRevision = 7 }; // Used by moc, qmetaobjectbuilder and qdbus

    int revision;
    int className;
    int classInfoCount, classInfoData;
    int methodCount, methodData;
    int propertyCount, propertyData;
    int enumeratorCount, enumeratorData;
    int constructorCount, constructorData; //since revision 2
    int flags; //since revision 3
    int signalCount; //since revision 4
  说明: 7,       // revision,就是将QMetaObjectPrivate中的revision复制为7.

转载地址:http://eidra.baihongyu.com/

你可能感兴趣的文章
第四节 对象和类
查看>>
闪迪(SanDisk)U盘防伪查询(官方网站)
查看>>
Android onMeasure方法介绍
查看>>
微信公众号搭建营销型房产项目程序后台开发
查看>>
git使用笔记
查看>>
无锁数据结构
查看>>
RabbitMQ消息队列:任务分发机制
查看>>
substr和substring的区别
查看>>
String.Format用法
查看>>
【转】java NiO 学习笔记
查看>>
MySQL的变量查看和设置
查看>>
Android NDK配、编译、调试
查看>>
长平狐 memcached源代码阅读笔记(二):网络处理部分
查看>>
android onNewIntent
查看>>
实战利用腾讯企业邮箱zabbix3.x邮件(微信/QQ/短信)告警详细配置
查看>>
干掉运营商:神奇盒子让你自建GSM 网络
查看>>
配置企业级wlan
查看>>
XML特殊符号
查看>>
kaptcha可配置项
查看>>
JavaMail邮箱验证用户注册
查看>>