Qt宏Q_OBJECT展开记录



  • c++中的宏

    #define A 3.14 展开后把A替换为B

    #define A(a) #a 展开后用a的字符串替换A(a)

    #define A(a) a##B 展开后把a和B的字符串连接起来后替换A(a)

    #include "common/common.h"
    #define PI 3.14
    #define PIPrivate 3.1415
    #define foo(arg) #arg
    #define bar(arg) arg##Private
    int main(int argc, char *argv[]) {
    
        DApplication::loadDXcbPlugin();
        DApplication app(argc, argv);
        Dtk::Util::DLogManager::registerConsoleAppender();
        Dtk::Util::DLogManager::registerFileAppender();
        qDebug()<<PI<<foo(PI)<<bar(PI)<<foo(bar(PI));
        return 0;
    }
    
    

    源码地址:GitHub

    在Qt中,用了不少的宏。最常见的就是Q_OBJECT

    Q_OBJECT展开的最终效果如下

    public:
        template<typename ThisObject>
        inline void qt_check_for_QOBJECT_macro(const ThisObject &_q_argument) const {
            int i = qYouForgotTheQ_OBJECT_Macro(this, &_q_argument);
            i = i + 1;
        }
        _Pragma(GCC diagnostic push)
        static const QMetaObject staticMetaObject;
        virtual const QMetaObject *metaObject() const;
        virtual void *qt_metacast(const char *);
        virtual int qt_metacall(QMetaObject::Call, int, void **);
        static inline QString tr(const char *s, const char *c = Q_NULLPTR, int n = -1) {
            return staticMetaObject.tr(s, c, n);
        }
    
        static inline QString trUtf8(const char *s, const char *c = Q_NULLPTR, int n = -1) {
            return staticMetaObject.tr(s, c, n);
        }
    private:
        __attribute__((visibility("hidden"))) static void qt_static_metacall(QObject *, QMetaObject::Call, int, void **);
        _Pragma(GCC diagnostic pop)
        struct QPrivateSignal {};
    

    下面做一步步展开
    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_CHECK展开如下,宏越来越多了。

    template<typename ThisObject>
    inline void qt_check_for_QOBJECT_macro(const ThisObject &_q_argument) const {
        int i = qYouForgotTheQ_OBJECT_Macro(this, &_q_argument);
        i = i + 1;
    }
    

    QT_WARNING_PUSHQT_WARNING_POP展开

    #define QT_WARNING_PUSH QT_DO_PRAGMA(GCC diagnostic push)
    #define QT_WARNING_POP QT_DO_PRAGMA(GCC diagnostic pop)
    

    QT_DO_PRAGMA展开

    #define QT_DO_PRAGMA(text)                      _Pragma(#text)
    

    又来了新的宏_Pragma,参考2.1.3 _Pragma操作符,就是告诉编译器GCC diagnostic push

    Q_OBJECT_NO_OVERRIDE_WARNING展开

    #if defined(Q_CC_CLANG) && Q_CC_CLANG >= 306
    #  define Q_OBJECT_NO_OVERRIDE_WARNING      QT_WARNING_DISABLE_CLANG("-Winconsistent-missing-override")
    #else
    #  define Q_OBJECT_NO_OVERRIDE_WARNING
    #endif
    
    

    QT_TR_FUNCTIONS展开,QT_DEPRECATED展开为空

    static inline QString tr(const char *s, const char *c = Q_NULLPTR, int n = -1) {
        return staticMetaObject.tr(s, c, n);
    }
    
    QT_DEPRECATED static inline QString
    trUtf8(const char *s, const char *c = Q_NULLPTR, int n = -1) {
        return staticMetaObject.tr(s, c, n);
    }
    

    Q_OBJECT_NO_ATTRIBUTES_WARNING展开,QT_WARNING_DISABLE_GCC展开为空

    #define Q_OBJECT_NO_ATTRIBUTES_WARNING QT_WARNING_DISABLE_GCC("-Wattributes")
    

    Q_DECL_HIDDEN_STATIC_METACALL展开,出现了__attribute__黑魔法,参考ATTRIBUTE 你知多少?

    #define Q_DECL_HIDDEN_STATIC_METACALL Q_DECL_HIDDEN
    #define Q_DECL_HIDDEN     __attribute__((visibility("hidden")))
    

    QT_ANNOTATE_CLASS展开为空

    #define QT_ANNOTATE_CLASS(type, ...)
    

    参考文章
    2.1.3 _Pragma操作符
    ATTRIBUTE 你知多少?

    源码地址:GitHub



  • @大黄老鼠 宏魔法真心黑啊



  • @大黄老鼠 因为GCC大家庭主要使用的是__attribute__修饰。
    而MSVC可以在类名或者函数名前面修饰。比如说__stdcall__declspec(dllexport)



  • @jiangcaiyang c++各种黑魔法,我怀恋java



  • @大黄老鼠Qt宏Q_OBJECT展开记录 中说:

    GitHub

    能变出钱来的魔法就是好魔法,人家用易语言的也有发了大财的。



  • @stlcours 支持一个。不管用什么语言,不断完善自己的产品才是正确的道路。


 

最近的回复

  • 0_1531879249244_9090909.jpg
    各位上午好哦。很高兴告诉大家,我们的萌梦社区又进行了一波更新了!
    这次主要更新在论坛代码层面上的,也就是说我们根据NodeBB官方的指示,将论坛从1.9.3升级到了1.10.1了!整个升级过程会产生一些服务暂停的现象,不过没有关系,很快就可以得到恢复,现在咱们可以顺利地在新的论坛上玩耍了!
    这次更新按照NodeBB官方的说法,主要在论坛的稳定性和迎合欧盟的GPDR政策(隐私权保护政策)上,也就是说,咱们论坛默认是支持用户的隐私权保护的~不要担心自己的隐私会被黑客利用什么的,但是大家还是要保护好平时论坛登录的密码哦。合适的时候还是使用浏览器的记住密码功能~

    此外,之前萌梦论坛之前的SSL加密是免费的,费时效果又不好。在手机浏览器上显示并非可靠的颁发者,现在换了新的证书,感觉好了不少。现在又可以在论坛上玩耍啦。

    好了,就是这样~

    阅读更多
  • @jiangcaiyang 是不支持QVector3D类型么,需要引用头文件还是需要转换类型呢

    阅读更多
  • 可能是Qt制作的Python绑定Shiboken遇到了问题。求先行者能够解忧~😵

    阅读更多
  • 0_1531804634749_5.jpg
    我使用的是PyCharm作为IDE,下载了Qt for Python(使用Qt 5.11),并且使用git克隆了pyside的源代码 + 例子。因为自己对OpenGL感兴趣,所以随手运行了一个例子。首先是PyOpenGL没有安装。

    0_1531804316021_1f365ea0-46b3-460d-9a00-c8cdeccf98af-image.png

    通过对话框弹出的命令,在terminal上运行

    python -m pip install PyOpenGL PyOpenGL_accelerate

    就可以了。

    虽然窗体以及3D内容出来了,但是命令提示符中会出现一些问题:

    D:/QtProject/pyside-setup/examples/opengl/hellogl2.py:394: RuntimeWarning: SbkConverter: Unimplemented C++ array type. self.program.setUniformValue(self.lightPosLoc, QVector3D(0, 0, 70)) D:/QtProject/pyside-setup/examples/opengl/hellogl2.py:423: RuntimeWarning: SbkConverter: Unimplemented C++ array type. self.program.setUniformValue(self.projMatrixLoc, self.proj) D:/QtProject/pyside-setup/examples/opengl/hellogl2.py:424: RuntimeWarning: SbkConverter: Unimplemented C++ array type. self.program.setUniformValue(self.mvMatrixLoc, self.camera * self.world) D:/QtProject/pyside-setup/examples/opengl/hellogl2.py:426: RuntimeWarning: SbkConverter: Unimplemented C++ array type. self.program.setUniformValue(self.normalMatrixLoc, normalMatrix)

    刚学习Python,还不清楚是怎么回事,怎么解决才好。

    阅读更多

关注我们

微博
QQ群











召唤伊斯特瓦尔