将私有子类添加到Q_DECLARE_METATYPE

时间:2019-07-07 15:46:08

标签: c++ qt qvariant

如何将MySubClass添加到Q_DECLARE_METATYPE。

这是示例代码,请忽略代码事务。

#include <QMetaType>
#include <QVariant>
#include <QDebug>

namespace MyNS {
    class MyClass : public QObject {
    public:
        MyClass() : value(0) { }
        MyClass(int value) : value(value) { }
        MyClass(const MyClass &other) { value = other.value; }
        ~MyClass() { }
        int getValue() const {
            MySubClass msc(value);
            QVariant v = QVariant::fromValue(msc);
            int tempV = v.value<MySubClass>().getSubValue();
            return tempV;
        }
    private:
        class MySubClass
        {
        public:
            MySubClass() : subValue(0) {}
            MySubClass(int v) : subValue(v)
            {

            }
            int getSubValue()
            {
                return subValue;
            }
        private:
            int subValue;
        };
//        Q_DECLARE_METATYPE(MySubClass);  error : 'QMetaTypeId<MyNS::MyClass::MySubClass>': cannot specialize template in current scope
        int value;
    };

//    Q_DECLARE_METATYPE(MySubClass);  error : 'MySubClass': undeclared identifier
}

Q_DECLARE_METATYPE(MyNS::MyClass);

int main(int argc, char *argv[])
{
    MyNS::MyClass m(15);
    QVariant v = QVariant::fromValue(m);
    qDebug() << v.canConvert<MyNS::MyClass>();
    qDebug() << v.value<MyNS::MyClass>().getValue();
}

考虑 MyClass 时,在没有子类的情况下效果很好,但是我的问题是子类。我想在代码中使用 MySubClass 和QVariant,但MySubClass是私有的。

1 个答案:

答案 0 :(得分:1)

要在您的QMetaType中使用MySubClass,它必须公开可用。

想想为什么要将MySubClass插入Qt的元类型系统中?

如果,您希望能够通过信号/插槽发送MySubClass,或对其进行反序列化/流化处理,则将使MySubClass成为公众兴趣。 The docs tell you so

  

在开始之前,我们需要确保所创建的自定义类型满足QMetaType提出的所有要求。换句话说,它必须提供:

a public default constructor,
a public copy constructor, and
a public destructor.

如果您只想将MySubClass实例存储在QVariant中,则可以编写自己的转换方法

QVariant toVariant() const

(static) MySubClass fromVariant(QVariant const &v)

为确保类型安全,您可以创建一些公共虚拟类型并在QMetaType中进行注册,以便获得唯一的ID,但是基本上您可以将MySubClass转换为{{1 }},然后返回,前提是所有包含的数据都可以存储在QVariantMap中。