在java语言中,可以使用getObject(String)函数,从类名直接构建新的对象。
而在C++中是没有这种机制的,Qt虽然提供了元对象机制,但只可以获取对象的类名,不能反向构建。
这个问题我在百度上找了很多,但都没有满意的答案,索性翻出去google一下,终于找到了解决方法。
原文地址:http://www.mimec.org/node/350
代码如下:
Reflect.h
#ifndef REFLECT_H
#define REFLECT_H
#include <QByteArray>
#include <QMetaObject>
#include <QHash>
class Reflect
{
public:
template<typename T>
static void registerClass()
{
constructors().insert( T::staticMetaObject.className(), &constructorHelper<T> );
}
static QObject* newInstance( const QByteArray& className, QObject* parent = nullptr )
{
Constructor constructor = constructors().value( className );
if ( constructor == nullptr )
return nullptr;
return (*constructor)( parent );
}
private:
typedef QObject* (*Constructor)( QObject* parent );
template<typename T>
static QObject* constructorHelper( QObject* parent )
{
return new T( parent );
}
static QHash<QByteArray, Constructor>& constructors()
{
static QHash<QByteArray, Constructor> instance;
return instance;
}
};
#endif // REFLECT_H
调用方法:
首先要注册你的class:(必须继承QObject并添加Q_OBJECT)
Reflect::registerClass<Print>();
反射构建一个新对象:
Print* object = (Print*)Reflect::newInstance("Print");
测试代码main.cpp:
#include <QCoreApplication>
#include "Reflect.h"
#include "print.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Reflect::registerClass<Print>();
Print* object = (Print*)Reflect::newInstance("Print");
object->func();
return a.exec();
}
print.h:
#ifndef PRINT_H
#define PRINT_H
#include <QObject>
#include <QDebug>
class Print : public QObject
{
Q_OBJECT
public:
Print(QObject* parent=nullptr)
:QObject(parent)
{
qDebug() << "constructor...";
}
void func()
{
qDebug() << "func...";
}
};
#endif // PRINT_H
结果:
作用:利用反射机制可以做很多事,例如,可以将反射与Qt的property相结合,实现hibernate框架(有空会再写一篇博客介绍)。
不足:这种实现方法不可避免地需要“先注册,后使用”,还达不到真正的反射,虽然可以在一个启动函数中注册所有的类,但还是比较麻烦的。
代码地址:https://gitee.com/bailiyang/cdemo/tree/master/Qt/35Reflect/
本文转载于:https://www.cnblogs.com/ztzheng/p/4122975.html
Qt实现反射其他参考链接:
《Qt 的反射(Reflection)应用》:https://blog.csdn.net/zyb408815041/article/details/87068291
《Qt论坛帖子》:https://forum.qt.io/topic/55761/solved-how-to-use-variadic-template-with-variadic-macro/5
《Qt自带的类反射机制》:https://blog.csdn.net/kenier/article/details/62041312