Qt中怎么根据类名来实例化对象
Qt中如何根据类名来实例化对象
QWidget* createWidget(const Qstring& name);
类似这个的一个函数,如何通过一个字符串函数名来实例一个具体的对象,求大牛
------解决方案--------------------
既然是这样,我就索性把这个问题,全部的回答一下。
对于Qt 来说,是可以做到运行时,根据对象的类名字(字符串)来获得对象的实例的,这点和一些语言的反射机制是一样的。
但是在Qt中,我们需要所额外的一步,就是注册。
只要做到了注册,我们就可以 *的创建对象了。
注册分为两步,定义的时候 使用 Q_DECLARE_METATYPE 宏。
使用的时候使用 qRegisterMetaType 函数
这样就可以在Qt 中动态地创建自己定义的类了。
但是这个方法,在注册 已有的 QWidget 和 QObject 的子类的时候会遇到问题。
原因是,在创建对象的时候,Qt 中有一个函数 qMetaTypeConstructHelper 会被调用,这个函数调用了欲创建类的复制构造函数,而复制构造函数 在 QWidget 和 QObject 的子类里是被禁用的。
解决的办法 就有两个,
一,启用 复制构造函数,
二,特化重写 qMetaTypeConstructHelper 函数。
复制构造函数 的启用 需要重新派生一个子类,就不多说了。
QWidget* createWidget(const Qstring& name);
类似这个的一个函数,如何通过一个字符串函数名来实例一个具体的对象,求大牛
------解决方案--------------------
既然是这样,我就索性把这个问题,全部的回答一下。
对于Qt 来说,是可以做到运行时,根据对象的类名字(字符串)来获得对象的实例的,这点和一些语言的反射机制是一样的。
但是在Qt中,我们需要所额外的一步,就是注册。
只要做到了注册,我们就可以 *的创建对象了。
#include <QtCore>
class Parser {
public:
virtual void parse() = 0;
virtual ~Parser() {}
};
class Parser1 : public Parser {
public:
Parser1() {
qDebug() <<"Parser1::Parser1()";
}
void parse() {
qDebug() << "Parser1::parse()";
}
~Parser1() {
qDebug() <<"Parser1::~Parser1()";
}
};
Q_DECLARE_METATYPE(Parser1)
class Parser2 : public Parser {
public:
Parser2() {
qDebug() <<"Parser2::Parser2()";
}
void parse() {
qDebug() << "Parser2::parse()";
}
~Parser2() {
qDebug() <<"Parser2::~Parser2()";
}
};
Q_DECLARE_METATYPE(Parser2)
void factory( const char* parserName ) {
int id = QMetaType::type( parserName );
if (id == -1) return; // ERROR HERE
Parser *parser = static_cast<Parser*>(QMetaType::construct(id));
parser->parse();
delete parser;
}
int main () {
qRegisterMetaType<Parser1>("Parser1");
qRegisterMetaType<Parser2>("Parser2");
qDebug() << "###### Trying create Parser1";
factory("Parser1");
qDebug() << "###### Trying create Parser2";
factory("Parser2");
}
注册分为两步,定义的时候 使用 Q_DECLARE_METATYPE 宏。
使用的时候使用 qRegisterMetaType 函数
这样就可以在Qt 中动态地创建自己定义的类了。
但是这个方法,在注册 已有的 QWidget 和 QObject 的子类的时候会遇到问题。
原因是,在创建对象的时候,Qt 中有一个函数 qMetaTypeConstructHelper 会被调用,这个函数调用了欲创建类的复制构造函数,而复制构造函数 在 QWidget 和 QObject 的子类里是被禁用的。
解决的办法 就有两个,
一,启用 复制构造函数,
二,特化重写 qMetaTypeConstructHelper 函数。
复制构造函数 的启用 需要重新派生一个子类,就不多说了。