QVariant中用了哪些技术

虽然,我也捋过源代码,还是这篇文章捋的更清楚,收录一下(你如果想看明白,打开源码,配合此篇文章撸) --- kangshifu

基本理解

Q_DECLARE_METATYPE
如果要使自定义类型或其他非QMetaType内置类型在QVaiant中使用,必须使用该宏。
该类型必须有公有的 构造、析构、复制构造 函数
qRegisterMetaType 必须使用该函数的两种情况
如果非QMetaType内置类型要在 Qt 的属性系统中使用
如果非QMetaType内置类型要在 queued 信号与槽 中使用

二者的代码:

Q_DECLARE_METATYPE 展开后是一个特化后的类 QMetaTypeId<TYPE>
qRegisterMetaType 将某类型注册中 MetaType 系统中

二者的联系:

QMetaTypeId<TYPE>的类中的成员包含对qRegisterMetaType的调用

我们知道类中的成员函数并不一定会被调用(即,该宏并不确保类型被注册到MetaType)。

通过qRegisterMetaType可以确保类型被注册
两个qRegisterMetaType 的联系

无参的qRegisterMetaType函数会通过该成员调用带参数的qRegisterMetaType()
这两个东西真难理清,不妨看看源码吧。

Q_DECLARE_METATYPE

代码来源:src/corelib/kernel/qmetatype.h

#define Q_DECLARE_METATYPE(TYPE)                                        \
    QT_BEGIN_NAMESPACE                                                  \
    template <>                                                         \
    struct QMetaTypeId< TYPE >                                          \
    {                                                                   \
        enum { Defined = 1 };                                           \
        static int qt_metatype_id()                                     \
            {                                                           \
                static QBasicAtomicInt metatype_id = Q_BASIC_ATOMIC_INITIALIZER(0); \
                if (!metatype_id)                                       \
                    metatype_id = qRegisterMetaType< TYPE >(#TYPE);     \
                return metatype_id;                                     \
            }                                                           \
    };                                                                  \
    QT_END_NAMESPACE

宏展开是一个在Qt的命名空间中的一个类模板的特化 QMetaTypeId<TYPE>

该类含一个enum和一个返回!QMetaType的id的成员函数
qRegisterMetaType(const char *typeName)

代码来源:src/corelib/kernel/qmetatype.h

template <typename T>
int qRegisterMetaType(const char *typeName)
{
    typedef void*(*ConstructPtr)(const T*);
    ConstructPtr cptr = qMetaTypeConstructHelper<T>;
    typedef void(*DeletePtr)(T*);
    DeletePtr dptr = qMetaTypeDeleteHelper<T>;

    return QMetaType::registerType(typeName, reinterpret_cast<QMetaType::Destructor>(dptr),
                                   reinterpret_cast<QMetaType::Constructor>(cptr));
}

该函数的核心就是调用了registerType 函数
两个Helper模板函数分别对构造和析构函数进行封装registerType

代码来源:src/corelib/kernel/qmetatype.cpp

int QMetaType::registerType(const char *typeName, Destructor destructor, Constructor constructor)

函数功能:

根据类型名查找其MetaType类型,如果已存在,则直接返回;否则创建后返回。

创建一个 !QCustomTypeInfo 对象
该对象包含要类型的构造、析构信息,已经规范化后的类型名
该对象存入一个全局的!QVector中
qRegisterMetaType()

看manual,可以知道,qRegisterMetaType 还有一个无参的重载函数。

template <typename T>
inline int qRegisterMetaType()
{
    return qMetaTypeId(static_cast<T *>(0));
}

函数看起来和带参数的那个似乎区别很大(难道不是么?)。
手册中告诉我们,执行这个的时候,模板参数T必须用 Q_DECLARE_METATYPE() 声明过
能猜到原因吗?注意看前面 Q_DECLARE_METATYPE() 代码,
对了。类中的成员函数qt_metatype_id中包含对qRegisterMetaType(typeName)的调用
这儿就是辗转调用了这个带参数的qRegisterMetaType函数

unregisterType(const char *typeName) //在Qt5中删除了

函数的作用是取消自己先前注册的某个metatype类型。

前面提到注册信息在一个全局的 QVector<QCustomTypeInfo>中,当取消注册的时候是怎么样的呢?直接删除Vector中相应的项么?源码告诉我们,不是的。

实际是查找到相应的项,清空该项的内容。

for (int v = 0; v < ct->count(); ++v)
 {
 if (ct->at(v).typeName == typeName)
 {
 QCustomTypeInfo &inf = (*ct)[v];
 inf.typeName.clear();
 inf.constr = 0;
 inf.destr = 0;
 inf.alias = -1;
 }
}

————————————————
版权声明:本文为CSDN博主「南方以北」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_25800311/article/details/88984223

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • C++音视频开发 面试1 技能要求:Socket,MFC,Windows 岗位职责:1、参与需求分析、模块开发等相...
    雪域迷影阅读 2,822评论 0 2
  • 简介 这里简单介绍Qt的一些核心机制,具体参见Qt文档。 主要包含内容: Qt的信号和槽,以及事件机制 Qt Ob...
    QuietHeart阅读 1,719评论 0 3
  • QT中文乱码 QTextCodec::setCodecForCStrings(QTextCodec::codecF...
    你猜_19ca阅读 1,071评论 0 1
  • Qt/QML 插件系统 本文将简要介绍一下 Qt 和 QML 的插件系统,并用几个简单的示例介绍 QML 的几种插...
    赵者也阅读 7,253评论 0 4
  • D-Pointer简介 如果你经常阅读QT的源码,你会看到大量的Q_D和Q_Q宏.这篇文章将会揭开这些宏的用处.这...
    托尼章阅读 1,151评论 1 1

友情链接更多精彩内容