在子类上调用虚拟方法将返回超类数据

Calling virtual method on subclass returns superclass data

本文关键字:返回 超类 数据 方法 虚拟 子类 调用      更新时间:2023-10-16

我有两个类,想要获取子类的元信息:

class DataObject : public QObject {
    Q_OBJECT
    ...
    public:
         virtual void meta() {
             const QMetaObject *mo = QObject::metaObject();
             for (int i = 0; i < mo->propertyCount(); i++) {
                 qDebug() << mo->property(i).name() << mo->property(i).read(this);
             }
         }
    ...
}
class User : public DataObject {
    Q_OBJECT
    Q_PROPERTY(int id MEMBER m_id)
    ...
}

当我在User对象上调用meta时,它只输出默认的对象属性objectName,而不是在User类中声明的属性。

我可以以某种方式获取子类的元信息,而无需手动实现每个子类的方法吗?

您需要

通过动态调度而不是静态调度来调用metaObject()Class::method()是静态调度)。你也不需要meta()是虚拟的,甚至不需要一个方法 - 它应该是一个独立的函数,它适用于你传递给它的任何对象;C++不是Java。

下面是一个示例。输出:

objectName QVariant(QString, "")
id QVariant(int, 1)
// https://github.com/KubaO/stackoverflown/tree/master/questions/meta-derived-38268004
#include <QtCore>
void meta(QObject * obj) {
   auto mo = obj->metaObject();
   for (int i = 0; i < mo->propertyCount(); i++)
      qDebug() << mo->property(i).name() << mo->property(i).read(obj);
}
struct DataObject : QObject {
   Q_OBJECT
};
struct User : DataObject {
   Q_PROPERTY(int id MEMBER m_id)
   Q_OBJECT
   int m_id { 1 };
};
int main() {
   User user;
   Q_ASSERT(user.metaObject()->propertyCount() == 2);
   meta(&user);
}
#include "main.moc"

您正在显式调用QObject::metaObject()方法,即QObject基类的方法。 从meta()实现中删除QObject::部分,以便在运行时调用metaObject()的虚拟实现