基于className的字符串获取QObject的子类的QMetaObject

时间:2017-10-28 17:46:48

标签: c++ qt

我正在编写一个加载插件的应用程序,每个插件都充当QObject子类的工厂。应用程序将参数发送到插件以实例化对象,每个对象使用不同的参数进行创建。

这一切都运行得很好,但是参数在插件和对象之间发生了变化,如果应用程序有一个模式来报告构造对象所需的参数,那就太好了。对象由配置文件创建,因此配置的作者需要知道要提供的必需参数,尤其是在它们发生更改时。

所以我一直在研究Qt的MetaObject系统,它非常适合这个用例,因为我可以使用Q_INVOKABLE创建一个报告对象构造函数形状的消息构造函数和QMetaObject :: constructor(int i)。

问题是我必须使用实际的类来获取QMetaObject静态实例,并且我想将类名列表作为字符串循环以创建此报告。

我对注册与QVariant一起使用的类型不感兴趣,因为这些对象不适合复制或通过网络发送或序列化。它们可能是连接到硬件或消耗大量内存的东西。我想要的只是插件工厂对象的构造函数定义,但源自插件可以构造的每个类的字符串名称。

QObject实例提供了一个QMetaObject,但它们似乎没有固有的QMetaType,这对于:

是必需的。
int typeId = QMetaType::type("MyClass");
const QMetaObject *s = QMetaType::metaObjectForType(typeId);

所以我必须使用

QMetaObject m = MyClass::staticMetaObject;

这需要将每个定义手动编码到插件中,如上所述:How can I find a Qt metaobject instance from a class name?

因为QObject没有复制构造函数,Q_DECLARE_METATYPE()将失败,以及qRegisterMetaType(),并且创建一个假的复制构造函数来满足编译器是一个坏主意,并且需要默认构造函数是坏的在这个用例中。

如果解决方案是手动将每个类键入到向配置编写器提供此消息的函数中,那么它就是这样。但是,如果我错过了一个更简单的解决方案,假设插件已经提供了一个可用对象的字符串列表,并作为QObject子类的工厂(没有默认构造函数),我想学习它!

有人更了解如何执行此操作:

QStringList p;
p << "MyClass";
p << "MyClass2";

for(int pi=0;pi<p.length();pi++)
{
    int typeId = QMetaType::type(p[pi]);
    const QMetaObject *s = QMetaType::metaObjectForType(typeId);
    int c = s->constructorCount();

    printf("constructor count: %d\n",c);

    for (int i=0;i<c;i++)
    {
        QMetaMethod m = s->constructor(i);

        printf("ctor %d: %s %s %d\n",i,qPrintable(m.name()),qPrintable(m.methodSignature()),m.parameterCount());

        int nc = m.parameterCount();
        QList<QByteArray> pn = m.parameterNames();
        QList<QByteArray> pt = m.parameterTypes();
        for (int j=0;j<nc;j++)
        {
            printf("\t%s %s\n",qPrintable(pt.at(j)),qPrintable(pn.at(j)));
        }
    }
}

这会因typeId == QMetaType::UnknownType而崩溃,因此s == null

1 个答案:

答案 0 :(得分:0)

最终,手动编码每个条目是我的解决方案。