Qt之QPluginLoader使用插件子项目及插件间通信(简易框架)(含部分源码+注释)

本文主要是介绍Qt之QPluginLoader使用插件子项目及插件间通信(简易框架)(含部分源码+注释),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、项目示例
    • 1.导航栏操作页面操作示例图
    • 2.打开所有页面操作示例图
    • 3.打开指定界面操作示例图
    • 3.插件重载操作演示
  • 二、插件逻辑个人理解
    • 1.QPluginLoader的简单使用
    • 2.子插件的基本要素
  • 三、项目结构(思路)简述
    • 1.定义插件接口类
    • 2.定义插件类别
      • 一个主项目
      • 若干子插件
    • 3.主项目及子插件的关联
  • 四、源码(此处列举主项目和一个子插件源码为例)
    • 1.主项目相关文件
      • iplugindatabusinterface.h
      • commondefins
      • mainwindow
      • main.cpp
    • 2.子插件(页面1)相关文件
      • pagefirstplugin
  • 总结

一、项目示例

1.导航栏操作页面操作示例图

下图演示了通过导航栏打开和关闭页面并在主页插件显示的操作。
在这里插入图片描述

2.打开所有页面操作示例图

下图演示了一键打开所有界面并同事更新导航来页面状态的操作。

在这里插入图片描述

3.打开指定界面操作示例图

下图演示了根据不同项按钮打开指定界面的操作,并且同时更新对应导航栏状态。
在这里插入图片描述

3.插件重载操作演示

下图演示了指定页面重载指定页面的逻辑演示。
注:本文代码此处重载并非同一插件的重载,而是将当前使用插件卸载,然后通过更新插件路径从而加载一个全新的插件逻辑。
在这里插入图片描述

二、插件逻辑个人理解

1.QPluginLoader的简单使用

使用步骤如下:

  1. **设置库文件:**创建一个QPluginLoader对象,可通过构造传参或setFileName函数设置库文件;
  2. **加载库文件:**使用QPluginLoader对象的load函数加载库文件;
  3. **获取插件指针:**通过QPluginLoader对象的instance函数获取一个QObject指针,然后通过转换获取,再使用插件指针获取控件对象等操作;
  4. 卸载插件:. 通过QPluginLoader对象的unload函数卸载插件,该函数会自动释放插件指针对象。

2.子插件的基本要素

  1. 继承自定义接口并根据功能实现对应的接口函数。
  2. 使用Q_DECLARE_INTERFACE:将接口对象声明给元对象,保证可使用Q_INTERFACES添加该接口。
  3. 使用Q_INTERFACES:添加后可使用qobject_cast转换为定义接口类(不添加该标识符,使用dynamic_cast强制转换也可使用,但不推荐)。
  4. 使用Q_PLUGIN_METADATA:添加后可使得QPluginLoader的instance函数获取到接口(类)指针。

三、项目结构(思路)简述

1.定义插件接口类

插件与插件之间无法直接通信,此时就需要一个接口作为中间类建立通信的桥梁(提供保障插件正常工作的函数,如收、发数据的函数),并且要求进行通信的子插件都需要继承接口类并实现对应的通信函数;以及在该类文件中定义唯一标识符(使得Qt能通过标识符识别该接口类),。

插件唯一标识符

// 通过宏定义插件标识符
#define InterfaceIdent "plugin.plugindatabusinterface"

插件相关必要函数

signals:// 数据通信信号void sigPluginCommTriggered(const StCommData &data);
public:// 数据接收处理函数virtual void recvPluginCommData(const StCommData &data) = 0;// 初始化函数virtual void initialize() = 0;// 初始化状态函数virtual bool isInitialized() const = 0;// 获取插件名称virtual QString name() const = 0;// 创建插件对象virtual QWidget *createWidget() = 0;
public slots:// 所有插件初始化完成函数virtual void slotInitialized() = 0;

2.定义插件类别

一个主项目

管理子插件集合以及负责各个子插件的互相通信数据转发;主项目负责加载、管理、卸载子插件并处理各个插件间的消息转发,并且主项目包含main.cpp,使得程序从该入口运行。

主项目的目录结构
在这里插入图片描述

若干子插件

某一功能的集合,负责该功能的消息发送及消息处理;要求继承接口类并个性化实现接口类虚函数,且子插件中的插件类中需添加指定的插件宏。

子插件接口宏使用

    // 添加后可使用qobject_cast转换为定义接口类(不添加该标识符,使用dynamic_cast强制转换也可使用,但不推荐)Q_INTERFACES(IPluginDataBusInterface)
#if QT_VERSION >= 0x050000// 添加后可使得QPluginLoader的instance函数获取到接口(类)指针Q_PLUGIN_METADATA(IID InterfaceIdent)
#endif // QT_VERSION >= 0x050000

子插件目录结构
在这里插入图片描述
其他子插件
除开初始化显示的导航栏插件和主页插件固定外,子页插件是通过导航栏插件解析配置文件获取,因此导航栏插件目录下添加其他插件信息配置文件。
如下图:
在这里插入图片描述

3.主项目及子插件的关联

以本文项目举例(详情请看源码
在主项目头文件中添加插件信息容器集合处理。
在这里插入图片描述
插件初始化时将对应插件存储至相关容器,并关联数据信号处理消息。
在这里插入图片描述

四、源码(此处列举主项目和一个子插件源码为例)

1.主项目相关文件

iplugindatabusinterface.h

#ifndef IPLUGINDATABUSINTERFACE_H
#define IPLUGINDATABUSINTERFACE_H#include <QObject>
#include "commondefins.h"// 通过宏定义插件标识符
#define InterfaceIdent "plugin.plugindatabusinterface"class IPluginDataBusInterface : public QObject
{Q_OBJECT
public:IPluginDataBusInterface(QObject *parent = Q_NULLPTR):QObject(parent){}signals:/*** @brief sigPluginCommTriggered 插件通信信号触发* @param data 插件数据*/void sigPluginCommTriggered(const StCommData &data);public:/*** @brief recvPluginCommData 接收插件数据信息* @param data 插件数据*/virtual void recvPluginCommData(const StCommData &data) = 0;/*** @brief initialize 初始化函数*/virtual void initialize() = 0;/*** @brief isInitialized 是否初始化* @return 初始化状态*/virtual bool isInitialized() const = 0;/*** @brief name 获取插件名* @return 插件名*/virtual QString name() const = 0;/*** @brief createWidget 创建插件控件* @param parent 插件控件父对象* @return 插件控件指针*/virtual QWidget *createWidget() = 0;public slots:/*** @brief slotInitialized 初始化完成槽函数*/virtual void slotInitialized() = 0;protected:bool m_initialized; // 是否初始化变量
};// 将接口对象声明给元对象
Q_DECLARE_INTERFACE(IPluginDataBusInterface, InterfaceIdent)#endif // IPLUGINDATABUSINTERFACE_H

commondefins

commondefins.h

#ifndef COMMONDEFINS_H
#define COMMONDEFINS_H#include <QHash>
#include <QPair>
#include <QString>
#include <QJsonObject>
#include <QMessageBox>#define DefaultPluginLoadPath QString("./") // 默认插件路径enum EmDataCode{PAGE_COMM = 0,PAGE_OPEN,PAGE_CLOSE,PAGE_RELOAD,PAGE_RELOAD_FAILED,PAGE_OPEN_ALL,PAGE_INIT,PAGE_INIT_FAILED,
};typedef struct StCommData {EmDataCode  code;       // 数据码QString     pageName;   // 页面名称QJsonObject commData;   // 通信数据StCommData(){}StCommData(EmDataCode code, QString pageName, QJsonObject commData) {this->code = code;this->pageName = pageName;this->commData = commData;}
}StCommData;// 初始化加载插件信息
typedef struct StInitPluginLoaderInfo {// 当运行模式为Debug时使用的区分使用
#ifdef QT_DEBUGconst QString navigationBar = DefaultPluginLoadPath + "navigationbarplugind.dll";const QString homePage = DefaultPluginLoadPath + "homepageplugind.dll";
#elseconst QString navigationBar = DefaultPluginLoadPath + "navigationbarplugin.dll";const QString homePage = DefaultPluginLoadPath + "homepageplugin.dll";
#endif
//    const QString homePage = DefaultPluginLoadPath + "pagefirstplugin.dll";
}StInitPluginLoaderInfo;// 主页加载插件信息
typedef struct StPluginCommInfo {const QStringList listNavigationBarInfo = {"PageFirst", "PageSecond"};  // 导航栏列表信息}StHomePagePluginLoaderInfo;#endif // COMMONDEFINS_H

commondefins.cpp

#include "commondefins.h"// 创建变量,保证项目可以全局使用
StInitPluginLoaderInfo g_stInitPluginLoaderInfo;

mainwindow

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H#include <QMainWindow>
#include <QPluginLoader>
#include "iplugindatabusinterface.h"namespace Ui {
class MainWindow;
}class MainWindow : public QMainWindow
{Q_OBJECTpublic:explicit MainWindow(QWidget *parent = Q_NULLPTR);~MainWindow();/*** @brief initial 初始化函数*/void initial();signals:/*** @brief sigInitialized 初始化完成信号*/void sigInitialized();private slots:/*** @brief slotPluginCommTriggered 插件通信槽函数* @param data 通信数据*/void slotPluginCommTriggered(const StCommData &data);private:Ui::MainWindow *ui;QPair<QPluginLoader *, IPluginDataBusInterface *>                   m_pairNavigateInfo;         // 导航栏指针信息QPair<QPluginLoader *, IPluginDataBusInterface *>                   m_pairHomePageInfo;         // 主页指针信息QHash<QString, QPair<QPluginLoader *, IPluginDataBusInterface *>>   m_hashPageLoaderIPlugin;    // 子页指针信息};#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"#include "commondefins.h"
#include <QDebug>
#include <QDateTime>
#include <QFile>extern StInitPluginLoaderInfo g_stInitPluginLoaderInfo;MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);initial();
}MainWindow::~MainWindow()
{// 卸载所有库m_pairNavigateInfo.first->unload();m_pairHomePageInfo.first->unload();foreach(auto info,m_hashPageLoaderIPlugin) {info.first->unload();}delete ui;
}void MainWindow::initial()
{// 初始化导航栏插件m_pairNavigateInfo.first = new QPluginLoader(g_stInitPluginLoaderInfo.navigationBar, this);m_pairNavigateInfo.second = qobject_cast<IPluginDataBusInterface *>(m_pairNavigateInfo.first->instance());connect(m_pairNavigateInfo.second, &IPluginDataBusInterface::sigPluginCommTriggered, this, &MainWindow::slotPluginCommTriggered);connect(this, &MainWindow::sigInitialized, m_pairNavigateInfo.second, &IPluginDataBusInterface::slotInitialized);ui->layoutNavigation->addWidget(m_pairNavigateInfo.second->createWidget());// 初始化主页插件m_pairHomePageInfo.first = new QPluginLoader(g_stInitPluginLoaderInfo.homePage, this);m_pairHomePageInfo.second = qobject_cast<IPluginDataBusInterface *>(m_pairHomePageInfo.first->instance());connect(m_pairHomePageInfo.second, &IPluginDataBusInterface::sigPluginCommTriggered, this, &MainWindow::slotPluginCommTriggered);connect(this, &MainWindow::sigInitialized, m_pairHomePageInfo.second, &IPluginDataBusInterface::slotInitialized);ui->tabWidget->addTab(m_pairHomePageInfo.second->createWidget(), u8"主页");emit sigInitialized();
}void MainWindow::slotPluginCommTriggered(const StCommData &data)
{switch (data.code) {case PAGE_INIT: {// 不同运行模式使用的库名不一样
#ifdef QT_DEBUGQString pluginName = data.commData.value("pluginName").toString() + "d.dll";
#elseQString pluginName = data.commData.value("pluginName").toString() + ".dll";
#endif// 指定加载库路径QPluginLoader *pageLoader = new QPluginLoader(DefaultPluginLoadPath + pluginName, this);// 链接页面信号IPluginDataBusInterface *pageInteface = qobject_cast<IPluginDataBusInterface *>(pageLoader->instance());// 直接判断接口对象,加载失败时指针为0x00,相当于falseif(pageInteface) {// 判断页面是否存在if(m_hashPageLoaderIPlugin.contains(data.pageName)) {QMessageBox::information(this, u8"提示", data.pageName + u8"页面名称已存在");break;}connect(pageInteface, &IPluginDataBusInterface::sigPluginCommTriggered, this, &MainWindow::slotPluginCommTriggered);// 添加页面指针信息m_hashPageLoaderIPlugin[data.pageName] = QPair<QPluginLoader *, IPluginDataBusInterface *>(pageLoader, pageInteface);m_pairHomePageInfo.second->recvPluginCommData(data);}else {StCommData reply = data;reply.code = PAGE_INIT_FAILED;QMessageBox::information(this, u8"提示", data.pageName + u8"页面信息初始化失败" + pageLoader->errorString());m_pairNavigateInfo.second->recvPluginCommData(reply);}break;}case PAGE_OPEN: {// 判断页面是否存在if(!m_hashPageLoaderIPlugin.contains(data.pageName)) {QMessageBox::information(this, u8"提示", data.pageName + u8"页面不存在");}// 判断页面打开状态else if (-1 != ui->tabWidget->indexOf(m_hashPageLoaderIPlugin[data.pageName].second->createWidget())) {QMessageBox::information(this, u8"提示", data.pageName + u8"页面已打开");}// 获取页面指针并打开页面else {QWidget *page = m_hashPageLoaderIPlugin[data.pageName].second->createWidget();ui->tabWidget->addTab(page, data.pageName);ui->tabWidget->setCurrentWidget(page);// 页面打开回执(原信息返回)m_pairNavigateInfo.second->recvPluginCommData(data);}break;}case PAGE_OPEN_ALL: {// 遍历容器,打开所有页面foreach(auto pageInfo, m_hashPageLoaderIPlugin) {ui->tabWidget->addTab(pageInfo.second->createWidget(), m_hashPageLoaderIPlugin.key(pageInfo));}// 页面关闭回执(原信息返回)m_pairNavigateInfo.second->recvPluginCommData(data);break;}case PAGE_CLOSE: {// 判断页面是否存在if (!m_hashPageLoaderIPlugin.contains(data.pageName)) {QMessageBox::information(this, u8"提示", data.pageName + u8"页面不存在");}// 判断页面打开状态else if (-1 == ui->tabWidget->indexOf(m_hashPageLoaderIPlugin[data.pageName].second->createWidget())) {QMessageBox::information(this, u8"提示", data.pageName + u8"页面暂未打开/已关闭");}// 关闭页面else{ui->tabWidget->removeTab(ui->tabWidget->indexOf(m_hashPageLoaderIPlugin[data.pageName].second->createWidget()));// 页面关闭回执(原信息返回)m_pairNavigateInfo.second->recvPluginCommData(data);}break;}case PAGE_RELOAD: {// 判断页面是否存在if (!m_hashPageLoaderIPlugin.contains(data.pageName)) {QMessageBox::information(this, u8"提示", data.pageName + u8"重载页面不存在");}else if(m_hashPageLoaderIPlugin[data.pageName].first->unload()) {QString fileName =  m_hashPageLoaderIPlugin[data.pageName].first->fileName().replace(DefaultPluginLoadPath, DefaultPluginLoadPath + "backup/");// 判断新文件名是否存在,存在即更新库名if (QFile::exists(fileName)) {m_hashPageLoaderIPlugin[data.pageName].first->setFileName(fileName);}// 加载库if (m_hashPageLoaderIPlugin[data.pageName].first->load()) {m_hashPageLoaderIPlugin[data.pageName].second = qobject_cast<IPluginDataBusInterface *>(m_hashPageLoaderIPlugin[data.pageName].first->instance());m_pairNavigateInfo.second->recvPluginCommData(data);m_hashPageLoaderIPlugin[data.pageName].second->createWidget();}else {StCommData tmpData = data;tmpData.code = PAGE_RELOAD_FAILED;m_pairNavigateInfo.second->recvPluginCommData(tmpData);QMessageBox::information(this, u8"提示", data.pageName + u8"页面重载失败 " + m_hashPageLoaderIPlugin[data.pageName].first->errorString());}}else {QMessageBox::information(this, u8"提示", data.pageName + u8"页面重载失败 " + m_hashPageLoaderIPlugin[data.pageName].first->errorString());}break;}default:break;}
}

main.cpp

#include "mainwindow.h"
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}

2.子插件(页面1)相关文件

注:该插件控件仅ui添加标识label,故仅展示插件文件

pagefirstplugin

pagefirstplugin.h

#ifndef PAGEFIRSTPLUGIN_H
#define PAGEFIRSTPLUGIN_H#include "iplugindatabusinterface.h"#include <QDesignerCustomWidgetInterface>class FormPageFirst;
class PageFirstPlugin : public IPluginDataBusInterface
{Q_OBJECTQ_INTERFACES(IPluginDataBusInterface)
#if QT_VERSION >= 0x050000Q_PLUGIN_METADATA(IID InterfaceIdent)
#endif // QT_VERSION >= 0x050000public:PageFirstPlugin(QObject *parent = Q_NULLPTR);~PageFirstPlugin();// IPluginDataBusInterface interface
public:void initialize();bool isInitialized() const;QString name() const;QWidget *createWidget();void recvPluginCommData(const StCommData &data);// IPluginDataBusInterface interface
public slots:void slotInitialized();private:FormPageFirst *m_formPageFirst = Q_NULLPTR;
};#endif // PAGEFIRSTPLUGIN_H

pagefirstplugin.cpp

#include "formpagefirst.h"
#include "pagefirstplugin.h"#include <QtPlugin>PageFirstPlugin::PageFirstPlugin(QObject *parent): IPluginDataBusInterface(parent)
{m_initialized = false;
}PageFirstPlugin::~PageFirstPlugin()
{if(Q_NULLPTR != m_formPageFirst) {delete m_formPageFirst;m_formPageFirst = Q_NULLPTR;}
}void PageFirstPlugin::initialize()
{if (m_initialized)return;// Add extension registrations, etc. herem_initialized = true;
}void PageFirstPlugin::slotInitialized()
{}void PageFirstPlugin::recvPluginCommData(const StCommData &data)
{}bool PageFirstPlugin::isInitialized() const
{return m_initialized;
}QWidget *PageFirstPlugin::createWidget()
{if(Q_NULLPTR == m_formPageFirst) {m_formPageFirst = new FormPageFirst();}return m_formPageFirst;
}QString PageFirstPlugin::name() const
{return QLatin1String("PageFirstPlugin");
}#if QT_VERSION < 0x050000
Q_EXPORT_PLUGIN2(pagefirstplugin, PageFirstPlugin)
#endif // QT_VERSION < 0x050000

总结

本文简易介绍插件使用、部分通信逻辑及插件构建逻辑,下一篇出插件创建详细步骤。(有需要可私源码)


友情提示——哪里看不懂可私哦,让我们一起互相进步吧
(创作不易,请留下一个免费的赞叭 谢谢 ^o^/)

注:文章为作者编程过程中所遇到的问题和总结,内容仅供参考,若有错误欢迎指出。
注:如有侵权,请联系作者删除


这篇关于Qt之QPluginLoader使用插件子项目及插件间通信(简易框架)(含部分源码+注释)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/775830

相关文章

SpringBoot中使用Flux实现流式返回的方法小结

《SpringBoot中使用Flux实现流式返回的方法小结》文章介绍流式返回(StreamingResponse)在SpringBoot中通过Flux实现,优势包括提升用户体验、降低内存消耗、支持长连... 目录背景流式返回的核心概念与优势1. 提升用户体验2. 降低内存消耗3. 支持长连接与实时通信在Sp

python使用库爬取m3u8文件的示例

《python使用库爬取m3u8文件的示例》本文主要介绍了python使用库爬取m3u8文件的示例,可以使用requests、m3u8、ffmpeg等库,实现获取、解析、下载视频片段并合并等步骤,具有... 目录一、准备工作二、获取m3u8文件内容三、解析m3u8文件四、下载视频片段五、合并视频片段六、错误

gitlab安装及邮箱配置和常用使用方式

《gitlab安装及邮箱配置和常用使用方式》:本文主要介绍gitlab安装及邮箱配置和常用使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装GitLab2.配置GitLab邮件服务3.GitLab的账号注册邮箱验证及其分组4.gitlab分支和标签的

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

nginx启动命令和默认配置文件的使用

《nginx启动命令和默认配置文件的使用》:本文主要介绍nginx启动命令和默认配置文件的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录常见命令nginx.conf配置文件location匹配规则图片服务器总结常见命令# 默认配置文件启动./nginx

在Windows上使用qemu安装ubuntu24.04服务器的详细指南

《在Windows上使用qemu安装ubuntu24.04服务器的详细指南》本文介绍了在Windows上使用QEMU安装Ubuntu24.04的全流程:安装QEMU、准备ISO镜像、创建虚拟磁盘、配置... 目录1. 安装QEMU环境2. 准备Ubuntu 24.04镜像3. 启动QEMU安装Ubuntu4

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

Python常用命令提示符使用方法详解

《Python常用命令提示符使用方法详解》在学习python的过程中,我们需要用到命令提示符(CMD)进行环境的配置,:本文主要介绍Python常用命令提示符使用方法的相关资料,文中通过代码介绍的... 目录一、python环境基础命令【Windows】1、检查Python是否安装2、 查看Python的安

Python并行处理实战之如何使用ProcessPoolExecutor加速计算

《Python并行处理实战之如何使用ProcessPoolExecutor加速计算》Python提供了多种并行处理的方式,其中concurrent.futures模块的ProcessPoolExecu... 目录简介完整代码示例代码解释1. 导入必要的模块2. 定义处理函数3. 主函数4. 生成数字列表5.