在我的 C++ Qt 应用程序(一个字典查找/测验程序)中,我使用插件为人们可能想要添加和使用的所有可能的语言提供功能。这些插件是在运行时从主窗口小部件的构造函数中搜索和加载的:
void MainForm::loadPlugins()
{
QDir curDir;
QStringList pluginFilter;
pluginFilter << "*_plugin.dll";
QStringList pluginFiles = curDir.entryList(pluginFilter, QDir::Files);
for (int i = 0; i < pluginFiles.size(); ++i)
{
const QString &pluginFile = pluginFiles.at(i);
QPluginLoader loader(pluginFile);
DictionaryPlugin *plug = qobject_cast<DictionaryPlugin*>(loader.instance());
if (plug)
{
plugins_.append(plug);
}
}
}
我曾经认为我应该自己销毁加载的插件,所以我迭代了plugins_
(指针的QList)~MainForm()
,依次删除它们。后来我从文档中QPluginLoader
发现我不应该这样做。此外,它还引起了一个奇怪的错误,导致程序在关闭时冻结(但仅在调试器下运行后),所以它显然有问题。
我从中删除了删除部分~MainForm()
,一切似乎都正常,除了我注意到插件的析构函数没有被调用。我依靠那些在应用程序终止时保存一些特定于插件的设置,否则我不会注意到。我在其中一个析构函数上放了一个断点,而调试器从未进入它。
以下是插件的概要:
// dict_plugin.h
class DictionaryPlugin
{
public:
virtual ~DictionaryPlugin() {}
virtual QString name() const = 0;
virtual QString language() const = 0;
/* ... */
};
Q_DECLARE_INTERFACE(DictionaryPlugin, "DictionaryPlugin")
// jp_plugin.h
class JpPlugin : public QObject, public DictionaryPlugin
{
Q_OBJECT
Q_PLUGIN_METADATA(IID "JpPlugin")
Q_INTERFACES(DictionaryPlugin)
public:
JpPlugin();
virtual ~JpPlugin();
virtual QString name() const { return QString("Japanese plugin v1.0"); }
virtual QString language() const { return QString("Japanese"); }
/* ... */
};
// jp_plugin.cpp
JpPlugin::~JpPlugin()
{
saveSettings();
delete dictWidget_;
delete settWidget_;
delete kanjiDialog_;
delete wordDialog_;
delete radicalDialog_;
}
我的问题是,插件是如何被销毁的?显然,当我的程序终止后操作系统回收内存时,它们必须如此。为什么省略了析构函数?还有其他方法可以让我释放他们吗?也许在类中保留原始QPluginLoader
对象MainForm
,然后unload()
在析构函数中手动保留它们?析构函数还负责销毁QDialog
插件拥有的一些 s 和 UI 小部件。这是否意味着程序退出时我有泄漏?
谢谢!
更新:QPluginLoader
我尝试为每个插件存储指向s 的指针。在~MainForm()
我迭代它们,调用unload()
和调用delete
它们。工作正常,除了冻结错误又回来了。当我关闭应用程序时,窗口消失了,但程序继续在调试器中运行。我可以暂停它,但堆栈只显示 ntdll.dll 中的一些身份不明的入口点。似乎没有立即发生任何不好的事情,正常运行时甚至都看不到,但不知何故让我感到不安。
UPDATE2:我让程序在冻结状态下运行(忘记停止它),发现它在一段时间后正常终止;它只花了大约 20 多秒的时间。现在我非常惊讶。什么可能导致这种情况?