我在使用GCC的Linux上使用Qt 4.8.3。我正在尝试使用QtPlugin创建模块化应用程序。该应用程序描述了一些基本功能,然后一系列插件将提供所有独特的功能。每个插件使用工厂提供类的多个实例,并包含工厂类和要提供的类。
目前,应用程序正在编译,但没有执行,我收到Cannot load library ~: (~: undefined symbol: _ZTI8Base16staticMetaObjectE)
等错误。我不知道如何解决这个错误。删除所有提到的Base修复了问题,但实例需要从该类派生,所以我认为这不会起作用。
我已经将可执行文件和单个插件包含在下面项目的结构中(它们目前都是相同的并且具有相同的问题)。
非常感谢您的协助。
main_project
+- main_project.pro
+- app/
| +- app.pro
| +- main.cpp
| +- factoryinterface.h
| +- base.h
| +- base.cpp
|
+- counter/
+- counter.pro
+- counter.h
+- counterfactory.cpp
+- counterfactory.h
+- counter.cpp
TEMPLATE = subdirs
SUBDIRS += \
app \
counter
QT += core
QT -= gui
DESTDIR = ../
TARGET = app_exec
CONFIG += console
CONFIG -= app_bundle
TEMPLATE = app
SOURCES += main.cpp \
base.cpp
HEADERS += \
base.h \
factoryinterface.h
#ifndef FACTORYINTERFACE_H
#define FACTORYINTERFACE_H
#include <QtPlugin>
#include "base.h"
class FactoryInterface {
public:
virtual ~FactoryInterface() {}
virtual Base *get() = 0;
virtual QString name() = 0;
};
Q_DECLARE_INTERFACE(FactoryInterface,
"com.example.FactoryInterface/1.0")
#endif // FACTORYINTERFACE_H
#ifndef BASE_H
#define BASE_H
#include <QObject>
#include <QString>
#include <QStringList>
#include <QHash>
class Base : public QObject {
Q_OBJECT
public:
Base(QObject *parent = 0);
virtual ~Base();
virtual QString name();
signals:
public slots:
protected:
QString name_;
};
#endif // BASE_H
#include "base.h"
Base::Base(QObject *parent) :
QObject(parent)
{
}
Base::~Base()
{
}
QString Base::name()
{
return name_;
}
#include <QtCore/QCoreApplication>
#include <QDir>
#include <QPluginLoader>
#include <QDebug>
#include "factoryinterface.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
QDir pluginDir(a.applicationDirPath());
pluginDir.cd("plugins");
foreach(QString filename, pluginDir.entryList(QDir::Files))
{
QPluginLoader loader(pluginDir.absoluteFilePath(filename));
QObject *plugin = loader.instance();
if (plugin)
{
FactoryInterface *i = qobject_cast<FactoryInterface*>(plugin);
if (i)
{
qDebug() << " :" << i->name();
}
} else {
qDebug() << loader.errorString();
}
}
return a.exec();
}
QT -= gui
CONFIG += plugin
TARGET = $$qtLibraryTarget(counter)
TEMPLATE = lib
DEFINES += COUNTER_LIBRARY
SOURCES += counter.cpp \
counterfactory.cpp
HEADERS += counter.h \
counterfactory.h
INCLUDEPATH += ../app
DESTDIR = ../plugins
symbian {
MMP_RULES += EXPORTUNFROZEN
TARGET.UID3 = 0xE5B38B75
TARGET.CAPABILITY =
TARGET.EPOCALLOWDLLDATA = 1
addFiles.sources = counter.dll
addFiles.path = !:/sys/bin
DEPLOYMENT += addFiles
}
unix:!symbian {
maemo5 {
target.path = /opt/usr/lib
} else {
target.path = /usr/lib
}
INSTALLS += target
}
#ifndef COUNTER_H
#define COUNTER_H
#include <QObject>
#include "base.h"
class Counter : public Base {
Q_OBJECT
public:
Counter();
};
#endif // COUNTER_H
#include "counter.h"
Counter::Counter()
{
name_ = "Counter";
}
#ifndef COUNTER_H
#define COUNTER_H
#include <QObject>
#include "factoryinterface.h"
#include "base.h"
#include "counter.h"
class CounterFactory : public QObject, public FactoryInterface
{
Q_OBJECT
Q_INTERFACES(FactoryInterface)
public:
explicit CounterFactory(QObject *parent = 0);
Base *get();
QString name();
signals:
public slots:
};
#endif // COUNTER_H
#include "counterfactory.h"
CounterFactory::CounterFactory(QObject *parent) :
QObject(parent)
{
}
Base *CounterFactory::get()
{
return new Counter;
}
QString CounterFactory::name()
{
return "Counter";
}
Q_EXPORT_PLUGIN2(counter, CounterFactory)
编辑:在调试模式下编译后更新了错误消息。
编辑:我尝试导出/导入BaseNode类,但这也不起作用。我不确定它是否完全不正确或我是否做错了。对Base类进行了以下更改(BASE_LIBRARY已添加到app.pro文件中的DEFINES):
#include <QtCore/qglobal.h>
#if defined(BASE_LIBRARY)
# define BASESHARED_EXPORT Q_DECL_EXPORT
#else
# define BASESHARED_EXPORT Q_DECL_IMPORT
#endif
class BASESHARED_EXPORT Base : public QObject {
答案 0 :(得分:0)
解决方案结果是将Base类添加到counter.pro文件中。我在下面列出了完整更新的counter.pro文件,但唯一的变化是添加到HEADERS和SOURCES。
QT -= gui
CONFIG += plugin
TARGET = $$qtLibraryTarget(counter)
TEMPLATE = lib
DEFINES += COUNTER_LIBRARY
SOURCES += counter.cpp \
counterfactory.cpp \
../app/base.cpp
HEADERS += counter.h \
counterfactory.h \
../app/base.h
INCLUDEPATH += ../app
DESTDIR = ../plugins
symbian {
MMP_RULES += EXPORTUNFROZEN
TARGET.UID3 = 0xE5B38B75
TARGET.CAPABILITY =
TARGET.EPOCALLOWDLLDATA = 1
addFiles.sources = counter.dll
addFiles.path = !:/sys/bin
DEPLOYMENT += addFiles
}
unix:!symbian {
maemo5 {
target.path = /opt/usr/lib
} else {
target.path = /usr/lib
}
INSTALLS += target
}