我正在构建Qt应用程序。虽然我之前编写了很多GUI的东西,并且框架本身没有出现问题,但我还没有编写任何能够加载插件的程序(在C#.NET应用程序之外)。我正在尝试构建的项目将插件作为其中不可或缺的一部分,我认为Qt框架看起来很不错,因为它几乎是普遍的跨平台,并且似乎有一个非常好的插件系统。
我的问题是:我似乎无法让QPluginLoader.instance()方法返回除空值之外的任何内容。
从我的读数来看,这是因为它找不到插件。我想我可能忘记做某事了,但我发现很少有关于实际编写插件的文档(有一些例子,但它们并不是非常详细)。有很多加载插件的例子,我认为我这样做是正确的,但我还没有真正找到一个实际制作插件的例子。
这就是我所做的:
使用Qt-Creator我创建了两个项目:一个窗口化的应用程序假装它是我的插件接收应用程序和一个共享库项目假装是插件
在我的窗口应用程序中,我有一个头文件,如下所示:
#ifndef PLUGININTERFACE_H
#define PLUGININTERFACE_H
#include <QtPlugin>
class QStringList;
class PluginInterface
{
public:
virtual ~PluginInterface() {};
virtual QStringList messages() const = 0;
};
Q_DECLARE_INTERFACE(PluginInterface,
"com.kevincuzner.LearningPlugins.PluginInterface/1.0")
#endif // PLUGININTERFACE_H
在我的共享库应用程序中,我创建了一个名为ATestPlugin的类(这也是项目的TARGET值):
#ifndef ATESTPLUGIN_H
#define ATESTPLUGIN_H
#include "ATestPlugin_global.h"
#include "../LearningPlugins/PluginInterface.h"
#include <QStringList>
class ATESTPLUGINSHARED_EXPORT ATestPlugin : public PluginInterface, public QObject
{
Q_OBJECT
Q_INTERFACES(PluginInterface)
public:
ATestPlugin();
virtual QStringList messages() const
{
//this part is actually defined in the .cpp file, but I don't feel like pasting that here
QStringList ret;
ret << "foo" << "bar" << "noms" << "Hello";
return ret;
}
};
#endif // ATESTPLUGIN_H
在ATestPlugin.cpp文件的末尾,我放置了Q_EXPORT_PLUGIN2(ATestPlugin, ATestPlugin)
然后在main方法的main.cpp文件中,我执行以下操作(&amp; w指向主窗口):
QString text = QFileDialog::getOpenFileName(&w, "Get a file");
QPluginLoader loader(text);
QObject* plugin = loader.instance();
if (plugin)
{
QMessageBox msgBox;
msgBox.setText(text);
msgBox.exec();
}
当我运行程序并从弹出的文件对话框中的库项目的构建目录中选择libATestPlugin.so时,我看不到消息框,这意味着* plugin为null。以前我总是弹出消息框并显示所选的文件名,所以我知道该部分正在运行。
有没有人想知道我需要做些什么才能让我的插件对QPluginLoader可见?
答案 0 :(得分:1)
它对我不起作用。将以前的TARGET更改为$$ qtLibraryTarget(...)可确保在不同平台上构建时的正确扩展。
我遇到了与QPluginLoader相同的问题。简单地说,我是如何解决它的:我创建了一个插件库,比如lib1.dll,它使用lib2.dll中的一些东西。在我的应用程序中,我尝试通过QPluginLoader加载lib1。
QPluginLoader loader( adaptersDir.absoluteFilePath(fileName) );
AdapterInterface* adapterIface = qobject_cast<AdapterInterface*>(loader.instance());
在这种情况下,loader.instance()返回0.解决方法是将lib2.dll复制到应用程序文件夹中,因为应用程序将其用于正确的加载插件lib1。
答案 1 :(得分:0)
经过进一步的实验,我设法让插件加载。我觉得这很羞怯,但这是RTFM的经典案例。
echo plugin tutorial(我在发布问题之后才找到)对库中的.pro文件进行了一些修改,以使其正确声明插件:
行TARGET = ATestPlugin
必须更改为TARGET = $$qtLibraryTarget(ATestPlugin)
。进行此更改后,上面程序中的消息框弹出,稍后当我运行qobject_cast< PluginInterface* >(plugin)
然后询问运行PluginInterface->messages()
时,我得到了在单独的类中实现的列表。
另外,我确实更改了继承顺序以将QObject放在第一位,因为它的工作原理我可能会这样做(尽管我不知道它是否有所不同)。