1

在我的 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 多秒的时间。现在我非常惊讶。什么可能导致这种情况?

4

1 回答 1

2

我在谷歌上找到了这个:

QPluginLoader 有自己的 ref 计数,并且会在为每个有句柄的 QPluginLoader 实例调用 unload() 时删除插件。在 QPluginLoader 析构函数中不会自动调用 unload(),因此永远不会自动删除插件。

它可以追溯到 2010 年,但也许您应该尝试保留对加载程序的引用并在程序结束时手动调用卸载。

这是文章的链接:

如何在 Qt 中卸载模块

于 2013-09-25T00:53:13.070 回答