【Qt之·Qt插件开发·(2)创建插件管理器统一管理插件】

系列文章目录

第一章 Qt插件创建并调用插件
第二章 Qt创建插件管理器统一管理插件
第三章 Qt插件之间相互通信
第四章 Qt创建并加载一个窗口插件
第五章 Qt插件工程作为子工程



前言


一、创建插件

1.插件接口基类

#include<QtPlugin>

class PluginInterface
{
public:
    virtual ~PluginInterface() {}
    virtual QString getName() const = 0;
    virtual QString showText() const = 0;
};

Q_DECLARE_INTERFACE(PluginInterface,"org.galaxyworld.plugins.PluginInterface/1.0")

2.插件

(1)插件1

#include "Plugin01_global.h"
#include<QObject>
#include"PluginInterface.h"

class PLUGIN01_EXPORT Plugin01 : public QObject , PluginInterface
{
    Q_OBJECT
    Q_INTERFACES(PluginInterface)
    Q_PLUGIN_METADATA(IID "plugin01")
public:
    Plugin01();
    virtual QString getName() const override
    {
        return "Plugin01";
    }

    virtual QString showText() const override
    {
        return "this is Plugin01";
    }
};

(2)插件2

#include "Plugin02_global.h"
#include<QObject>
#include"PluginInterface.h"

class PLUGIN02_EXPORT Plugin02 : public QObject , PluginInterface
{
    Q_OBJECT
    Q_INTERFACES(PluginInterface)
    Q_PLUGIN_METADATA(IID "plugin02")
public:
    Plugin02();
    virtual QString getName() const override
    {
        return "Plugin02";
    }

    virtual QString showText() const override
    {
        return "this is Plugin02";
    }
};

二、插件管理器

class PluginManger : public QWidget
{
    Q_OBJECT

public:

    static PluginManger *instance()
    {
        if(nullptr == m_instance)
        {
            m_instance = new PluginManger;
        }
        return m_instance;
    }

    void loadAllPlugins();
    void unloadAllPlugins();
    void loadPlugin(const QString &filePath);
    void unloadPlugin(const QString &filePath);
    QPluginLoader *getPlugin(const QString &name);
    QVariant getPluginName(QPluginLoader *loader);

private:
    explicit PluginManger(QObject *parent = nullptr);
    ~PluginManger();

    QHash<QString,QPluginLoader *> loaders;
    QHash<QString,QString> names;

    static PluginManger *m_instance;

    class GarbageCollector
    {
        ~GarbageCollector()
        {
            if(PluginManger::instance())
            {
                delete  PluginManger::instance();
                PluginManger::m_instance = nullptr;
            }
        }
    };

    static GarbageCollector gc;

};
#include <QDir>
#include<QDebug>
#include<QFileInfo>

PluginManger* PluginManger::m_instance;

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    PluginManger *p = PluginManger::instance();

    p->loadAllPlugins();

}


Widget::~Widget()
{
    delete ui;
}


void PluginManger::loadAllPlugins()
{
    QDir pluginsdir(QDir::currentPath());
    qDebug() << QDir::currentPath();
    pluginsdir.cd("..\\Lib");

    QFileInfoList pluginsInfo = pluginsdir.entryInfoList(\
                QDir::Files | QDir::NoDotAndDotDot);

    for(QFileInfo fileInfo : pluginsInfo)
            loadPlugin(fileInfo.absoluteFilePath());
}

void PluginManger::unloadAllPlugins()
{
    for(QString filePath : loaders.keys())
        unloadPlugin(filePath);
}

void PluginManger::loadPlugin(const QString &filePath)
{
    if(!QLibrary::isLibrary(filePath))
        return;

    QPluginLoader *loader = new QPluginLoader(filePath);
    QString pluginName;
    if(loader->load())
    {
        PluginInterface *plugin = qobject_cast<PluginInterface *>(loader->instance());
        if(plugin)
        {
            pluginName = plugin->getName();
            loaders.insert(filePath,loader);
            names.insert(filePath,pluginName);
            qDebug() << "插件名称:" << plugin->getName() << "插件信息" << plugin->showText();
        }else {
            delete loader;
            loader = nullptr;

        }

    }else {
        qDebug() << "loadPlugin:" << filePath << loader->errorString();
    }
}

void PluginManger::unloadPlugin(const QString &filePath)
{

    QPluginLoader *loader = loaders.value(filePath);
    if(loader->unload())
    {
        loaders.remove(filePath);
        delete loader;
        loader = nullptr;
    }
}

PluginManger::PluginManger(QObject *parent)
{

}

PluginManger::~PluginManger()
{
    unloadAllPlugins();
}

总结

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值