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

Get QMetaObject for a subclass of a QObject based on a string of the className

本文关键字:子类 获取 QObject QMetaObject className 字符串 基于      更新时间:2023-10-16

我正在编写一个加载插件的应用程序,每个插件都充当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;

这需要将每个定义手动编码到插件中,如前所述:如何从类名中找到Qt元对象实例?

由于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: %dn",c);
for (int i=0;i<c;i++)
{
QMetaMethod m = s->constructor(i);
printf("ctor %d: %s %s %dn",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 %sn",qPrintable(pt.at(j)),qPrintable(pn.at(j)));
}
}
}

这将崩溃,因为typeId == QMetaType::UnknownType,因此s == null

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