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

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

git使用的说明总结

Git使用说明 下载安装(下载地址) macOS: Git - Downloading macOS Windows: Git - Downloading Windows Linux/Unix: Git (git-scm.com) 创建新仓库 本地创建新仓库:创建新文件夹,进入文件夹目录,执行指令 git init ,用以创建新的git 克隆仓库 执行指令用以创建一个本地仓库的