天天看點

Qt Core學習日記——第一天Moc檔案

作者:嗷嗚笨

深度

本文主要分析Windows平台,Mac、Linux暫不涉及

本文隻分析到Win32 API/Windows Com元件/STL庫函數層次,再下層代碼不做探究

本文QT版本5.15.2

Moc

頭檔案XTest.h:

#pragma once

#include <qobject.h>

class XTest : public QObject

{

Q_OBJECT

Q_CLASSINFO("author", "Sabrina Schweinsteiger")

public:

XTest(QObject* p = nullptr);

~XTest();

public slots:

void slot1(double * p1);

Q_INVOKABLE void slot2(int * p2);

Q_REVISION(1) void slot3(char * p3);

signals:

void sig1(void * p4);

};

cpp檔案XTest.cpp:

/****************************************************************************

** Meta object code from reading C++ file 'XTest.h'

**

** Created by: The Qt Meta Object Compiler version 67 (Qt 5.15.2)

**

** WARNING! All changes made in this file will be lost!

*****************************************************************************/

#include <memory>

#include "../../../XTest.h"

#include <QtCore/qbytearray.h>

#include <QtCore/qmetatype.h>

#if !defined(Q_MOC_OUTPUT_REVISION)

#error "The header file 'XTest.h' doesn't include <QObject>."

#elif Q_MOC_OUTPUT_REVISION != 67

#error "This file was generated using the moc from 5.15.2. It"

#error "cannot be used with the include files from this version of Qt."

#error "(The moc has changed too much.)"

#endif

QT_BEGIN_MOC_NAMESPACE

QT_WARNING_PUSH

QT_WARNING_DISABLE_DEPRECATED

struct qt_meta_stringdata_XTest_t {

QByteArrayData data[15];

char stringdata0[91];

};

#define QT_MOC_LITERAL(idx, ofs, len) \

Q_STATIC_BYTE_ARRAY_DATA_HEADER_INITIALIZER_WITH_OFFSET(len, \

qptrdiff(offsetof(qt_meta_stringdata_XTest_t, stringdata0) + ofs \

- idx * sizeof(QByteArrayData)) \

)

static const qt_meta_stringdata_XTest_t qt_meta_stringdata_XTest = {

{

QT_MOC_LITERAL(0, 0, 5), // "XTest"

QT_MOC_LITERAL(1, 6, 6), // "author"

QT_MOC_LITERAL(2, 13, 22), // "Sabrina Schweinsteiger"

QT_MOC_LITERAL(3, 36, 4), // "sig1"

QT_MOC_LITERAL(4, 41, 0), // ""

QT_MOC_LITERAL(5, 42, 2), // "p4"

QT_MOC_LITERAL(6, 45, 5), // "slot1"

QT_MOC_LITERAL(7, 51, 7), // "double*"

QT_MOC_LITERAL(8, 59, 2), // "p1"

QT_MOC_LITERAL(9, 62, 5), // "slot2"

QT_MOC_LITERAL(10, 68, 4), // "int*"

QT_MOC_LITERAL(11, 73, 2), // "p2"

QT_MOC_LITERAL(12, 76, 5), // "slot3"

QT_MOC_LITERAL(13, 82, 5), // "char*"

QT_MOC_LITERAL(14, 88, 2) // "p3"

},

"XTest\0author\0Sabrina Schweinsteiger\0"

"sig1\0\0p4\0slot1\0double*\0p1\0slot2\0int*\0"

"p2\0slot3\0char*\0p3"

};

#undef QT_MOC_LITERAL

static const uint qt_meta_data_XTest[] = {

// content:

8, // revision

0, // classname

1, 14, // classinfo

4, 16, // methods

0, 0, // properties

0, 0, // enums/sets

0, 0, // constructors

0, // flags

1, // signalCount

// classinfo: key, value

1, 2,

// signals: name, argc, parameters, tag, flags

3, 1, 40, 4, 0x06 /* Public */,

// slots: name, argc, parameters, tag, flags

6, 1, 43, 4, 0x0a /* Public */,

9, 1, 46, 4, 0x0a /* Public */,

12, 1, 49, 4, 0x8a /* Public | MethodRevisioned */,

// signals: revision

0,

// slots: revision

0,

0,

1,

// signals: parameters

QMetaType::Void, QMetaType::VoidStar, 5,

// slots: parameters

QMetaType::Void, 0x80000000 | 7, 8,

QMetaType::Void, 0x80000000 | 10, 11,

QMetaType::Void, 0x80000000 | 13, 14,

0 // eod

};

void XTest::qt_static_metacall(QObject *_o, QMetaObject::Call _c, int _id, void **_a)

{

if (_c == QMetaObject::InvokeMetaMethod) {

auto *_t = static_cast<XTest *>(_o);

Q_UNUSED(_t)

switch (_id) {

case 0: _t->sig1((*reinterpret_cast< void*(*)>(_a[1]))); break;

case 1: _t->slot1((*reinterpret_cast< double*(*)>(_a[1]))); break;

case 2: _t->slot2((*reinterpret_cast< int*(*)>(_a[1]))); break;

case 3: _t->slot3((*reinterpret_cast< char*(*)>(_a[1]))); break;

default: ;

}

} else if (_c == QMetaObject::IndexOfMethod) {

int *result = reinterpret_cast<int *>(_a[0]);

{

using _t = void (XTest::*)(void * );

if (*reinterpret_cast<_t *>(_a[1]) == static_cast<_t>(&XTest::sig1)) {

*result = 0;

return;

}

}

}

}

QT_INIT_METAOBJECT const QMetaObject XTest::staticMetaObject = { {

QMetaObject::SuperData::link<QObject::staticMetaObject>(),

qt_meta_stringdata_XTest.data,

qt_meta_data_XTest,

qt_static_metacall,

nullptr,

nullptr

} };

const QMetaObject *XTest::metaObject() const

{

return QObject::d_ptr->metaObject ? QObject::d_ptr->dynamicMetaObject() : &staticMetaObject;

}

void *XTest::qt_metacast(const char *_clname)

{

if (!_clname) return nullptr;

if (!strcmp(_clname, qt_meta_stringdata_XTest.stringdata0))

return static_cast<void*>(this);

return QObject::qt_metacast(_clname);

}

int XTest::qt_metacall(QMetaObject::Call _c, int _id, void **_a)

{

_id = QObject::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<int*>(_a[0]) = -1;

_id -= 4;

}

return _id;

}

// SIGNAL 0

void XTest::sig1(void * _t1)

{

void *_a[] = { nullptr, const_cast<void*>(reinterpret_cast<const void*>(std::addressof(_t1))) };

QMetaObject::activate(this, &staticMetaObject, 0, _a);

}

QT_WARNING_POP

QT_END_MOC_NAMESPACE

QT_BEGIN_MOC_NAMESPACE

如果沒有定義QT_NAMESPACE或者定義了Q_MOC_RUN則為空

否則為using namespace ::QT_NAMESPACE

預設QT_BEGIN_MOC_NAMESPACE為空

QT_WARNING_PUSH

預設定義__pragma(warning(push)),存儲每個警告的目前警告狀态

QT_WARNING_DISABLE_DEPRECATED

__pragma(warning(disable: 4996))關閉編譯警告4996

qt_meta_stringdata_XTest_t

定義資料結構用于存儲中繼資料資訊,包括類名稱、函數名稱

data、stringdata0并不是恒定大小,依照頭檔案定義函數、參數數量決定

QT_MOC_LITERAL

用于指定函數名稱的位置

#define QT_MOC_LITERAL(idx, ofs, len) \

{ { {-1} }, len, 0, 0, qptrdiff(((::size_t)&reinterpret_cast<char const volatile&>((((qt_meta_stringdata_XTest_t*)0)->stringdata0))) + ofs - idx * sizeof(QByteArrayData))}

qt_meta_stringdata_XTest

存儲目前類的中繼資料資訊

static const qt_meta_stringdata_XTest_t qt_meta_stringdata_XTest = {

{

QT_MOC_LITERAL(0, 0, 5), // "XTest"

QT_MOC_LITERAL(1, 6, 6), // "author"

QT_MOC_LITERAL(2, 13, 22), // "Sabrina Schweinsteiger"

QT_MOC_LITERAL(3, 36, 4), // "sig1"

QT_MOC_LITERAL(4, 41, 0), // ""

QT_MOC_LITERAL(5, 42, 2), // "p4"

QT_MOC_LITERAL(6, 45, 5), // "slot1"

QT_MOC_LITERAL(7, 51, 7), // "double*"

QT_MOC_LITERAL(8, 59, 2), // "p1"

QT_MOC_LITERAL(9, 62, 5), // "slot2"

QT_MOC_LITERAL(10, 68, 4), // "int*"

QT_MOC_LITERAL(11, 73, 2), // "p2"

QT_MOC_LITERAL(12, 76, 5), // "slot3"

QT_MOC_LITERAL(13, 82, 5), // "char*"

QT_MOC_LITERAL(14, 88, 2) // "p3"

},

"XTest\0author\0Sabrina Schweinsteiger\0"

"sig1\0\0p4\0slot1\0double*\0p1\0slot2\0int*\0"

"p2\0slot3\0char*\0p3"

};

将宏翻譯後得

static const qt_meta_stringdata_XTest_t qt_meta_stringdata_XTest = {

{

QT_MOC_LITERAL(0, 0, 5), // "XTest"

{-1,5,0,0,0x90},//是對QByteArrayData的定義

QT_MOC_LITERAL(1, 6, 6), // "author"

QT_MOC_LITERAL(2, 13, 22), // "Sabrina Schweinsteiger"

QT_MOC_LITERAL(3, 36, 4), // "sig1"

QT_MOC_LITERAL(4, 41, 0), // ""

QT_MOC_LITERAL(5, 42, 2), // "p4"

QT_MOC_LITERAL(6, 45, 5), // "slot1"

QT_MOC_LITERAL(7, 51, 7), // "double*"

QT_MOC_LITERAL(8, 59, 2), // "p1"

QT_MOC_LITERAL(9, 62, 5), // "slot2"

QT_MOC_LITERAL(10, 68, 4), // "int*"

QT_MOC_LITERAL(11, 73, 2), // "p2"

QT_MOC_LITERAL(12, 76, 5), // "slot3"

QT_MOC_LITERAL(13, 82, 5), // "char*"

QT_MOC_LITERAL(14, 88, 2) // "p3"

},

"XTest\0author\0Sabrina Schweinsteiger\0"

//類名稱後接classinfo定義‘\0’隔開

"sig1\0\0p4\0slot1\0double*\0p1\0slot2\0int*\0"

"p2\0slot3\0char*\0p3"

};

QT_MOC_LITERAL(0, 0, 5),定義的是"XTest\0sig1\0\0slot1\0slot2\0slot3"中"XTest"的位置,類型為QByteArray.替換宏後為{-1,5,0,0,0x90},-1是QByteArray的引用計數,5為QByteArrayData的大小,0為配置設定位址,0為預留容量,0x90為"XTest\0author\0Sabrina Schweinsteiger\0"中"XTest"的首位置

假若h檔案中函數改為

void slot1(double * p1);

Q_INVOKABLE void slot2(int * p2);

Q_REVISION(1) void slot3(char * p3);

qt_meta_stringdata_XTest變化如下:

static const qt_meta_stringdata_XTest_t qt_meta_stringdata_XTest = {

{

QT_MOC_LITERAL(0, 0, 5), // "XTest"

QT_MOC_LITERAL(1, 6, 4), // "sig1"

QT_MOC_LITERAL(2, 11, 0), // ""

QT_MOC_LITERAL(3, 12, 5), // "slot1"

QT_MOC_LITERAL(4, 18, 7), // "double*"

QT_MOC_LITERAL(5, 26, 2), // "p1"

QT_MOC_LITERAL(6, 29, 5), // "slot2"

QT_MOC_LITERAL(7, 35, 4), // "int*"

QT_MOC_LITERAL(8, 40, 2), // "p2"

QT_MOC_LITERAL(9, 43, 5), // "slot3"

QT_MOC_LITERAL(10, 49, 5), // "char*"

QT_MOC_LITERAL(11, 55, 2) // "p3"

},

"XTest\0sig1\0\0slot1\0double*\0p1\0slot2\0"

"int*\0p2\0slot3\0char*\0p3"

};

每個函數QT_MOC_LITERAL(3, 12, 5), // "slot1"定義下面緊接QT_MOC_LITERAL(4, 18, 7), // "double*"參數類型,類型後面接參數名稱QT_MOC_LITERAL(5, 26, 2), // "p1"

qt_meta_data_XTest

static const uint qt_meta_data_XTest[] = {

// content:

8, // revision

0, // classname

0, 0, // classinfo

4, 14, // methods

0, 0, // properties

0, 0, // enums/sets

0, 0, // constructors

0, // flags

1, // signalCount

// signals: name, argc, parameters, tag, flags

1, 0, 38, 2, 0x06 /* Public */,

// slots: name, argc, parameters, tag, flags

3, 1, 39, 2, 0x0a /* Public */,

6, 1, 42, 2, 0x0a /* Public */,

9, 1, 45, 2, 0x8a /* Public | MethodRevisioned */,

// signals: revision

0,

// slots: revision

0,

0,

1,

// signals: parameters

QMetaType::Void,

// slots: parameters

QMetaType::Void, 0x80000000 | 4, 5,

QMetaType::Void, 0x80000000 | 7, 8,

QMetaType::Void, 0x80000000 | 10, 11,

0 // eod

};

3, 1, 39, 2, 0x0a /* Public */,3代表QT_MOC_LITERAL(3, 12, 5), // "slot1"函數,1參數數量,39代表參數“p1”位置,0x0a代表函數屬性

XTest::qt_static_metacall

主要對外函數調用,消息循環觸發過來時通過此函數調用到槽函數

QMetaObject *XTest::metaObject

擷取QObject中繼資料QMetaObject

寫到最後

各位看官有什麼意見或建議可以寫到評論區,大家一起探讨

繼續閱讀