一、插件开发入门【Qt环境-mingw6.5.3-qmake版】-封装dll调用

news2025/1/20 6:01:13

1.先创建一个名为mainProject的主项目,并进行简单的ui布局

image.png
image.png
image.png
image.png
image.png

2.使用C++ Library库创建动态库项目作为主项目的子插件

image.png

2.1 Qt模块选择为Widgets

image.png

2.2 环境配置和主项目要一致

image.png

3.给插件添加qt界面设计类

image.png
image.png

image.png
image.png

4.给subPlugin插件添加一个名为PluginInterface.h的头文件

image.png
image.png

#ifndef PLUGININTERFACE_H
#define PLUGININTERFACE_H

#include <QWidget>

#define PluginInterfaceIID "com.szweebon.PluginInterface/1.0.0" // 定义一个插件接口的唯一标识符(IID),用于标识插件的接口版本

class PluginInterface{ // 这是一个纯虚基类(接口),用于定义插件接口的标准。
public:
    virtual ~PluginInterface() = default; // 虚析构函数,确保派生类在销毁时能够正确释放资源
    virtual QWidget *getWidget() = 0;   // 该函数用于返回一个 QWidget 指针,表示插件提供的界面部件
};

Q_DECLARE_INTERFACE(PluginInterface, PluginInterfaceIID) // 确保接口类和它的 IID 在 Qt 元对象系统中注册
#endif // PLUGININTERFACE_H

5.编辑subplugin.h和subplugin.cpp代码

#ifndef SUBPLUGIN_H
#define SUBPLUGIN_H
#include "PluginInterface.h"
#include "subPlugin_global.h"

class SUBPLUGIN_EXPORT SubPlugin: public QObject, public PluginInterface
{
    Q_OBJECT
    Q_PLUGIN_METADATA(IID PluginInterfaceIID)//PluginInterfaceIID就是IID
    Q_INTERFACES(PluginInterface)
public:
    SubPlugin();
    ~SubPlugin();
    QWidget *getWidget();
private:
    QWidget *mGenWidget;
};

#endif // SUBPLUGIN_H
#include "subplugin.h"
#include "form.h"
SubPlugin::SubPlugin() {

    mGenWidget = Q_NULLPTR;
}

SubPlugin::~SubPlugin() {

    if(mGenWidget){
        mGenWidget->deleteLater();
    }
}
QWidget *SubPlugin::getWidget()
{
    if(!mGenWidget){
        mGenWidget = new Form;
    }
    return mGenWidget;
}

6.去掉mainProject项目和subPlugin项目的的影子编译

image.png
image.png

7.将插件subPlugin项目的PluginInterface.h赋值一份到mianProject项目中,并添加到主项目录

image.png
image.png
image.png

8.把subPlugin项目生成的dll文件复制一份到mianProject主项目

这个运行动态库项目没有exe是很正常的。反正我的没有,有dll就行
image.png
image.png

image.png

9.编写主程序代码

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>

#include <QPluginLoader>//插件加载类

QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private slots:
    void on_pushButton_clicked();

private:
    Ui::MainWindow *ui;
    QPluginLoader  loader;
    bool isPluginLoaded;         // 标志插件是否已加载
    void removeTabWithTitle(const QString &title);
};
#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "PluginInterface.h"
#include <QDir>
#include <QMessageBox>

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
    , isPluginLoaded(false)  // 新增标志以跟踪插件是否已加载
{
    ui->setupUi(this);
    // 初始化按钮文本
    ui->pushButton->setText("加载插件");
}

MainWindow::~MainWindow()
{
    // 在销毁窗口时,检查并卸载已加载的插件
    if (isPluginLoaded) {
        loader.unload();
    }
    delete ui;
}

void MainWindow::on_pushButton_clicked()
{
    if (isPluginLoaded) {
        // 如果插件已经加载,执行卸载操作
        loader.unload();
        removeTabWithTitle("插件标签");  // 清除具有特定标题的标签页
        isPluginLoaded = false;  // 更新插件加载标志
        ui->pushButton->setText("加载插件");  // 更新按钮文本
        qDebug("插件已卸载");
    } else {
        // 插件未加载时,进行加载操作
        QDir dir("D:/EmbeddedDevelopment/Qt/Plugins/mainProject/debug");
        // 如果需要使用应用程序的运行目录,可以启用以下代码
        // QDir dir(qApp->applicationDirPath());

        qDebug("目录1: %s", dir.path().toUtf8().constData());  // 输出目录路径

        // 设置插件文件路径并准备加载插件
        loader.setFileName(dir.filePath("subPlugin.dll"));
        qDebug("目录2: %s", dir.filePath("subPlugin.dll").toUtf8().constData());  // 输出插件文件路径

        // 尝试加载插件
        if (!loader.load()) {
            qDebug("Failed to load loader");
            QMessageBox::critical(this, "错误", loader.errorString());  // 显示错误信息
            return;
        }

        // 创建插件实例
        QObject *plugin = loader.instance();
        if (!plugin) {  // 如果插件实例为空,输出错误信息并退出
            qDebug("loader.instance() is null");
            return;
        }
        qDebug("插件实例创建成功");

        // 将插件实例转换为 PluginInterface 类型
        PluginInterface *pi = qobject_cast<PluginInterface *>(plugin);
        if (pi) {  // 如果转换成功,获取插件的 QWidget 并添加到 TabWidget 中
            QWidget *w = pi->getWidget();
            if (w) {
                ui->tabWidget->addTab(w, "插件标签");  // 添加插件的 QWidget 为新标签页
                qDebug("插件标签添加成功");
                isPluginLoaded = true;  // 更新插件加载标志
                ui->pushButton->setText("卸载插件");  // 更新按钮文本
            } else {
                qDebug("插件未返回有效的 QWidget");
            }
        } else {
            qDebug("插件转换失败,pi is NULL");
        }
    }
}

void MainWindow::removeTabWithTitle(const QString &title)
{
    int count = ui->tabWidget->count();
    for (int i = 0; i < count; ++i) {
        if (ui->tabWidget->tabText(i) == title) {
            ui->tabWidget->removeTab(i);
            return;  // 一旦找到并移除目标标签页后退出
        }
    }
}

10.运行结果

可以添加卸载插件界面,其实添加的是一个不能自己独立运行的封装好的动态库界面,而且项目直接没有层级,故我不建议采纳这种方法
image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2059446.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

软件设计师全套备考系列文章8 -- 查找、排序

软考-- 软件设计师&#xff08;8&#xff09;-- 查找、排序 文章目录 软考-- 软件设计师&#xff08;8&#xff09;-- 查找、排序前言一、查找二、排序三、排序的评价指标&#xff08;重点&#xff09; 前言 考试时间&#xff1a;每年5月、11月&#xff0c;软件设计师每年都会…

湖南 | 产能利用率高达80%的龙头砼企如何进行站内外高效配合

走进标杆企业 走进标杆企业&#xff0c;感受名企力量&#xff0c;探寻学习优秀企业领先之道。本期要跟砼行们推介的标杆企业是湖南省新宁县鑫旺混凝土有限公司。 鑫旺混凝土有限公司成立于2012年&#xff0c;建有三条混凝土生产线&#xff0c;自有水泥、砂石骨料、房地产等一…

vue3 中 defineProps 和 defineEmits

在 Vue 3 中&#xff0c;defineProps 和 defineEmits 是组合式 API 的核心功能&#xff0c;用于处理父子组件之间的传值和事件通信。 1. defineProps defineProps 用于定义并接收父组件传递过来的数据&#xff08;props&#xff09;。它是在子组件中使用的&#xff0c;接收的…

100000在银行存个3年利息居然12000

python存款利息问题 设计一个函数来计算存款利息。存款利息由存款金额和存款时间决定。 如果存款金额小于或等于5000元&#xff0c;则年利率为2%;5000元到10000元之间&#xff0c;则年利率为3%;大于10000元&#xff0c;则年利率为4%。返回本金和利息 def getMoney(amount,yea…

44.开发商购买土地

44.开发商购买土地 &#xff08;用到了前缀和的知识&#xff09; 题目链接 //卡码网题号44.开发商购买土地 #include<iostream> #include<vector> #include<climits> using namespace std;int main() {int n, m;int sum0;cin >> n >> m;vector…

stable-diffusion-webui容器构建教程

一、介绍 Stable Diffusion WebUI 是一个提供了易于使用的 AI 绘画工具&#xff0c;它允许用户通过一个更友好、可视化的网页界面来与 Stable Diffusion 模型互动&#xff0c;可以实现文生图、图生图等。 二、特点 易于使用的界面 &#xff1a;用户可以通过网页界面进行操作…

【数据同步】SeaTunnel初体验,5000字深入浅出带你用上Oracle-CDC

Apache SeaTunnel 是啥&#xff1f;下一代高性能、分布式、海量数据集成框架。支持上百个数据源、传输速度快、准确率高&#xff0c;丰富易扩展的连接器和插件化的连接器设计&#xff0c;能够更轻松的运行复杂的集成。是一个分布式、高性能的数据集成平台&#xff0c;用于数据迁…

USB Type-C如何取9V、12V、15V、20V电压-PD快充协议芯片ECP5701

相信大家在生活中也发现了&#xff0c;现在越来越多的设备都改用这种type-C接口的母座进行取电了。 因为欧盟决议 &#xff1a;自2024年起部分消费电子产品必须提供单一的USB-C充电接口。 那么这种type-C接口相比之前的Micro-B接口有着一个很大的优势就是可以有更高的电压&…

部署同步工具syncthing

1、下载包arm包&#xff08;根据自己的环境下载包&#xff09; #进到指定目录 cd /usr/local/ #可以根据自己的环境下载不同版本的包 wget https://github.com/syncthing/syncthing/releases/download/v1.27.10/syncthing-linux-arm64-v1.27.10.tar.gz2、进行部署 #将其解压 …

接口参数与文档||关于淘宝商品·订单数据API接口的功能达成经验分享

电商数据采集有5种方式&#xff0c;包括API、RPA、数据库连接、Excel下载和ERP等业务系统数据采集。这些方法可帮助卖家获取多平台电商数据&#xff0c;进行深度挖掘&#xff0c;实现电商运营的优化。 电商竞争白热化的今天&#xff0c;一个电商卖家往往会在多个平台铺设店铺来…

Flutter Web 正式官宣弃用 HTML renderer , Canvas 路线成为唯一

Flutter Web 团队计划在 2025 年的第一个 Flutter stable 版本中弃用 HTML renderer&#xff0c;当然在 master 和 beta 中会更早合并这一更改。 关于这个话题&#xff0c;其实在年初的我就曾发布过 《Flutter 即将放弃 Html renderer 》&#xff0c; Html renderer 从 2018 年…

狗都能看懂的Swin Transformer的讲解和代码实现

文章目录 1、Swin-Transformer介绍2、模型整体框架3、Patch Mergeing详解4、W-MSA模块详解MSA模块计算量W-MSA模块计算量 5、SW-MSA详解6、Relative Position Bias详解7、模型详细配置参数 1、Swin-Transformer介绍 自从ViT&#xff08;Vision Transformer&#xff09;出现之后…

腾讯软件测试岗二面:web 测试问题被虐哭了,直到学长给了我这些知识点.....

web 测试一直是大厂软件测试问到的一个重点&#xff0c;下面给大家展示下大厂关于web 测试经常会问到的一些问题&#xff0c;以及解析。看当面试官问到你这些问题的时候&#xff0c;你是否也能够对答如流。 web 测试面试真题及解析&#xff1a; 一&#xff0c;描述用浏览器访问…

SQL基础——函数与约束

声明&#xff1a;以下内容为根据黑马数据库视频教程&#xff0c;个人整理的笔记&#xff0c;方便记录学习。 SQL基础之函数与约束 知识导图案例SQL语句编写一、函数1.字符串函数2.数值函数3.日期函数4.流程函数 二、约束 知识导图 案例SQL语句编写 一、函数 1.字符串函数 A.…

用工业操作系统鸿道Intewell可以玩黑神话悟空吗?

黑神话悟空的爆火&#xff0c;让我想到&#xff0c;工业操作系统鸿道Intewell是否可以玩黑神话悟空&#xff1f; 鸿道Intewell操作系统是面向工业控制领域的操作系统&#xff0c;它支持实时和非实时应用在同一个硬件平台上运行&#xff0c;并且能够保留Windows开发环境的同时部…

平衡二叉树(AVLTree)

1.平衡二叉树的定义 1.1 什么是平衡二叉树 平衡二叉树&#xff0c;又称AVL树&#xff0c;用于解决二叉排序树高度不确定的情况&#xff0c;如果二叉排序树的子树间的高度相差太大&#xff0c;就会让二叉排序树操作的时间复杂度升级为O(n)&#xff0c;为了避免这一情况&#x…

mac 安装Arthas

mac安装有两种方式 1.第一步安装Arthas 第一种&#xff1a; curl -L https://arthas.aliyun.com/install.sh | sh 第二种jar包形式 curl -O https://arthas.aliyun.com/arthas-boot.jar个人比较推荐第一种因为运行测试成功了 第一种安装后可能会出现一些命令不符合 需…

NVIDIA刚刚发布了关于 AI 的免费在线课程!

英伟达最近上线了一批不错的免费课程&#xff0c;先收藏起来&#xff01; ps:有时候真想有一个收藏即学会的技能啊 数据中心中的人工智能 了解数据中心的 AI 基础知识&#xff0c;涵盖机器学习、深度学习、GPU 架构和部署。 了解多系统 AI 集群和基础设施规划。 课程地址&…

勇闯机器学习(第三关-特征工程)

以下内容皆为原创&#xff0c;制作不易&#xff0c;请帅锅、镁铝点点赞赞和关注吧❥(^_^) 一.提问环节 机器学习是什么&#xff1f; 机器学习就是通过自动分析大量数据去建立模型&#xff0c;训练模型&#xff0c;预测数据。 这么好记的概念&#xff0c;你应该记住了吧&#x…

多线程编程的拙见

一. 线程和进程的概念 1.为什么引入多线程编程&#xff1f; 在多线程&#xff08;Multithreaded&#xff0c;MT&#xff09;编程出现之前&#xff0c;电脑程序的运行由一个执行序列组成&#xff0c;执行序列按顺序在主机的中央处理器CPU中运行。即使整个程序由多个相互独立无…