创建插件 DLL 项目
Step 1: 创建插件 DLL 项目
在 Visual Studio 中创建一个新的 DLL 项目,并添加以下文件和代码。
头文件:CShapeBase.h
cpp
复制代码
#pragma once #include <afxwin.h> // MFC 必需头文件 #include <string> #include <vector> #include "tinyxml.h" // 使用 TiXml 库的 XML 支持 #include "CxImage.h" // 使用 CxImage 库的图像处理支持 enum EDrawShape { RECTANGLE, CIRCLE, LINE }; // 示例绘制形状类型枚举 class CShapeBase { public: CShapeBase(EDrawShape nDrawShape); virtual ~CShapeBase(); virtual void Draw(CDC* pDC) = 0; virtual void InertRootItem(TiXmlElement* ItemElement, CString sItem, CString str); virtual void InertRootItem(TiXmlElement* ItemElement, CString sItem, int value); virtual void InertRootItem(TiXmlElement* ItemElement, CString sItem, float value); virtual void LoadDetPara(TiXmlElement* pDetElement); virtual void OnDrawResult(CDC* pDC); EDrawShape GetShapeType(); protected: EDrawShape m_nDrawShape; std::vector<CPoint> m_points; };
源文件:CShapeBase.cpp
cpp
复制代码
#include "CShapeBase.h" CShapeBase::CShapeBase(EDrawShape nDrawShape) : m_nDrawShape(nDrawShape) {} CShapeBase::~CShapeBase() {} void CShapeBase::InertRootItem(TiXmlElement* ItemElement, CString sItem, CString str) { ItemElement->SetAttribute(std::string(CT2A(sItem)).c_str(), std::string(CT2A(str)).c_str()); } void CShapeBase::InertRootItem(TiXmlElement* ItemElement, CString sItem, int value) { ItemElement->SetAttribute(std::string(CT2A(sItem)).c_str(), value); } void CShapeBase::InertRootItem(TiXmlElement* ItemElement, CString sItem, float value) { ItemElement->SetDoubleAttribute(std::string(CT2A(sItem)).c_str(), value); } void CShapeBase::LoadDetPara(TiXmlElement* pDetElement) { // 加载检测参数的示例实现 } void CShapeBase::OnDrawResult(CDC* pDC) { // 示例绘制结果 } EDrawShape CShapeBase::GetShapeType() { return m_nDrawShape; }
派生类:CSetDetDamage.h
cpp
复制代码
#pragma once #include "CShapeBase.h" class CSetDetDamage : public CShapeBase { public: CSetDetDamage(); void Draw(CDC* pDC) override; void OnDrawResult(CDC* pDC) override; };
派生类实现:CSetDetDamage.cpp
cpp
复制代码
#include "CSetDetDamage.h" CSetDetDamage::CSetDetDamage() : CShapeBase(RECTANGLE) {} void CSetDetDamage::Draw(CDC* pDC) { CRect rect(10, 10, 100, 100); pDC->Rectangle(rect); } void CSetDetDamage::OnDrawResult(CDC* pDC) { // 定义特定的绘图逻辑,例如检测结果 }
导出函数的实现:Plugin.cpp
cpp
复制代码
#include "pch.h" #include "CSetDetDamage.h" extern "C" __declspec(dllexport) CShapeBase* CreateShapeInstance() { return new CSetDetDamage(); } extern "C" __declspec(dllexport) void DestroyShapeInstance(CShapeBase* pInstance) { delete pInstance; } extern "C" __declspec(dllexport) EDrawShape GetShapeType(CShapeBase* pInstance) { return pInstance->GetShapeType(); }
主程序示例代码
在主程序中,可以通过动态加载 DLL,调用导出的创建和销毁函数。
cpp
复制代码
#include <afxwin.h> #include <Windows.h> #include <iostream> #include "CShapeBase.h" typedef CShapeBase* (*CreateShapeFunc)(); typedef void (*DestroyShapeFunc)(CShapeBase*); typedef EDrawShape (*GetShapeTypeFunc)(CShapeBase*); class CMainApp : public CWinApp { public: BOOL InitInstance() override { CWnd wnd; wnd.Create(NULL, _T("Main Window")); HMODULE hModule = LoadLibrary(L"YourPlugin.dll"); if (!hModule) { std::cerr << "Failed to load DLL" << std::endl; return FALSE; } CreateShapeFunc createShape = (CreateShapeFunc)GetProcAddress(hModule, "CreateShapeInstance"); DestroyShapeFunc destroyShape = (DestroyShapeFunc)GetProcAddress(hModule, "DestroyShapeInstance"); GetShapeTypeFunc getShapeType = (GetShapeTypeFunc)GetProcAddress(hModule, "GetShapeType"); if (createShape && destroyShape && getShapeType) { CShapeBase* pShape = createShape(); CDC* pDC = wnd.GetDC(); pShape->Draw(pDC); EDrawShape shapeType = getShapeType(pShape); std::cout << "Shape type: " << shapeType << std::endl; destroyShape(pShape); wnd.ReleaseDC(pDC); } FreeLibrary(hModule); return TRUE; } }; CMainApp theApp;
编译和运行步骤
- 插件 DLL:编译并生成
YourPlugin.dll文件。 - 主程序:通过
LoadLibrary动态加载 DLL,并调用导出函数创建CSetDetDamage实例,绘制形状并获取形状类型。
注意事项
- DLL 和主程序的字符集(如 Unicode 或 ANSI)应保持一致。
- 使用
tinyxml和CxImage库时,请确保库文件正确配置。 - 为提高健壮性,可以在 DLL 导出函数中加入错误检查和异常处理逻辑。
4o
您说:
如果有N个插件如何实现和管理
ChatGPT 说:
ChatGPT
记忆已更新
要管理和实现多个插件,可以采用插件管理器(Plugin Manager)模式。该模式能够动态加载、卸载、并调用不同的插件,以下是一个实现方法:
1. 插件接口定义
定义一个通用接口,所有插件需要实现这个接口。可以创建一个基类 IPlugin,包含每个插件应实现的虚函数:
cpp
复制代码
class IPlugin { public: virtual ~IPlugin() {} virtual void Initialize() = 0; // 初始化插件 virtual void Execute() = 0; // 执行插件操作 virtual void Shutdown() = 0; // 关闭插件 };
2. 插件管理器类
创建 PluginManager 类来管理多个插件。该类可以动态加载 DLL、存储已加载插件的实例,并调用相应的接口方法。
cpp
复制代码
#include <vector> #include <string> #include <Windows.h> class PluginManager { public: void LoadPlugin(const std::string& dllPath); void UnloadPlugins(); void ExecuteAllPlugins(); private: struct PluginHandle { HMODULE hModule; IPlugin* instance; }; std::vector<PluginHandle> plugins; }; void PluginManager::LoadPlugin(const std::string& dllPath) { HMODULE hModule = LoadLibrary(dllPath.c_str()); if (hModule) { typedef IPlugin* (*CreatePluginFunc)(); auto CreatePlugin = (CreatePluginFunc)GetProcAddress(hModule, "CreatePluginInstance"); if (CreatePlugin) { IPlugin* plugin = CreatePlugin(); plugin->Initialize(); plugins.push_back({hModule, plugin}); } else { FreeLibrary(hModule); } } } void PluginManager::UnloadPlugins() { for (auto& plugin : plugins) { plugin.instance->Shutdown(); delete plugin.instance; FreeLibrary(plugin.hModule); } plugins.clear(); } void PluginManager::ExecuteAllPlugins() { for (auto& plugin : plugins) { plugin.instance->Execute(); } }
3. 插件 DLL 实现
插件 DLL 文件中实现 IPlugin 接口,并提供 CreatePluginInstance 导出函数。
cpp
复制代码
#include "IPlugin.h" class SamplePlugin : public IPlugin { public: void Initialize() override { // 初始化代码 } void Execute() override { // 执行插件的主要功能 } void Shutdown() override { // 资源清理 } }; extern "C" __declspec(dllexport) IPlugin* CreatePluginInstance() { return new SamplePlugin(); }
4. 插件的加载和管理
在主程序中,创建 PluginManager 的实例并加载所需的插件:
cpp
复制代码
int main() { PluginManager manager; manager.LoadPlugin("PluginA.dll"); manager.LoadPlugin("PluginB.dll"); manager.ExecuteAllPlugins(); manager.UnloadPlugins(); return 0; }
管理多个插件
- 每个插件都通过路径加载,插件管理器可以灵活管理插件的数量。
ExecuteAllPlugins()函数可以调用所有加载的插件方法,可以根据需要增加更复杂的管理操作,例如按插件类型调用或异步执行。
4o
您说:
#include "CShapeBase.h"这类头文件,主程序和dll都是一个文件,是吗
ChatGPT 说:
ChatGPT
在插件架构中,通常 主程序和插件 DLL会共享某些头文件,比如接口类(如 IPlugin)和基类(如 CShapeBase)的定义,以确保两者之间的接口和数据结构保持一致。
为实现共享:
- 公共头文件:将
CShapeBase.h等通用接口头文件放在一个公共的包含目录中,主程序和 DLL 项目都可以访问该目录中的文件。 - 主程序和 DLL 的分离:主程序和 DLL 不应在同一个源文件中实现。一般而言:
- 主程序:包含插件管理器(
PluginManager),并动态加载 DLL。 - DLL:实现插件的功能,并包含从公共头文件中继承的接口和基类(如
CShapeBase的实现)。
- 主程序:包含插件管理器(
目录结构示例
less
复制代码
ProjectRoot/ ├── include/ │ ├── CShapeBase.h // 通用接口和基类头文件 │ └── IPlugin.h // 插件接口定义 ├── main_program/ // 主程序代码目录 │ ├── PluginManager.cpp │ └── main.cpp └── plugins/ // 插件代码目录 ├── PluginA/ │ ├── PluginA.cpp │ └── PluginA.def // 导出符号定义 └── PluginB/ ├── PluginB.cpp └── PluginB.def // 导出符号定义
在这种结构下:
- 头文件:主程序和插件都可以包含
include/目录中的公共头文件。 - 编译:主程序和 DLL 项目独立编译,但共享接口定义和基类头文件。
- 加载:主程序通过插件管理器加载不同 DLL 插件的实例。
编译配置
- 主程序:将
include/目录配置为包含目录。 - DLL 项目:也需包含
include/目录,并实现导出的工厂函数(如CreatePluginInstance())。
这样设置后,主程序和 DLL 都可以访问和使用 CShapeBase 类的定义,实现独立编译但共享结构的插件架构。
优点
- 代码解耦:将
CDrawTool放入 DLL 中可以减少主程序的代码量,使代码更易维护。 - 便于更新:如果需要对
CDrawTool类进行修改或更新,可以仅更新 DLL,而不必重新编译整个主程序。 - 扩展性:将绘图功能放在插件中,可以方便地为主程序添加更多功能,例如不同类型的绘图工具等。
实现要点
将 CDrawTool 类放到 DLL 中后,主程序需要使用动态链接库来加载和调用该类。实现中需要注意以下几点:
-
导出类:在 DLL 中将
CDrawTool类标记为导出(__declspec(dllexport)),以便主程序可以访问该类。cpp
复制代码
#ifdef DRAWTOOL_EXPORTS #define DRAWTOOL_API __declspec(dllexport) #else #define DRAWTOOL_API __declspec(dllimport) #endif class DRAWTOOL_API CDrawTool { // 类定义 }; -
动态加载和接口设计:在主程序中,可以使用工厂函数(例如
CreateDrawToolInstance)来动态创建CDrawTool实例。工厂函数需要导出,以便主程序可以通过该函数获得CDrawTool的实例。 -
对象传递:确保主程序将
CWnd、CDC等窗口资源传递给 DLL 中的CDrawTool实例,这样它就可以在主程序的窗口上绘图。 -
依赖管理:主程序和插件之间的类和数据结构应尽量保持一致,例如
CShapeBase类和EDrawShape枚举等。
调用示例
假设 DLL 中导出一个工厂函数:
cpp
复制代码
extern "C" DRAWTOOL_API CDrawTool* CreateDrawToolInstance();
主程序可以这样使用:
cpp
复制代码
typedef CDrawTool* (*CreateDrawToolFunc)(); HINSTANCE hInst = LoadLibrary("DrawTool.dll"); CreateDrawToolFunc createFunc = (CreateDrawToolFunc)GetProcAddress(hInst, "CreateDrawToolInstance"); CDrawTool* pDrawTool = createFunc(); pDrawTool->SetDrawWnd(pMainWnd); // 传入主窗口实例 // 使用其他方法
这样,CDrawTool 类的所有实现都在插件 DLL 中完成,主程序无需直接依赖类的具体实现。
4o
您说:
那主界面class CCVideoWnd : public CStatic类里面如何使用CDrawTool它
ChatGPT 说:
ChatGPT
记忆已更新
要在主界面类 CCVideoWnd 中使用 CDrawTool(如果 CDrawTool 已移到插件 DLL 中),可以通过以下方式实现:
-
加载 DLL 和获取
CDrawTool实例:在CCVideoWnd中,动态加载包含CDrawTool的 DLL,并通过导出函数来创建CDrawTool对象的实例。 -
使用导出函数创建对象:假设 DLL 中有一个导出函数,例如
CreateDrawTool,可以通过该函数创建CDrawTool的实例。cpp
复制代码
typedef CDrawTool* (*LPFN_CREATEDRAWTOOL)(); HMODULE hModule = LoadLibrary(_T("YourPluginDLL.dll")); if (hModule) { LPFN_CREATEDRAWTOOL pfnCreateDrawTool = (LPFN_CREATEDRAWTOOL)GetProcAddress(hModule, "CreateDrawTool"); if (pfnCreateDrawTool) { m_pDrawTool = pfnCreateDrawTool(); } } -
调用
CDrawTool成员函数:在CCVideoWnd中可以像操作普通对象一样调用CDrawTool的方法,但要确保CDrawTool的接口是公开的且可通过 DLL 边界调用。 -
释放资源:在
CCVideoWnd的析构函数中销毁CDrawTool实例,并释放 DLL。cpp
复制代码
if (m_pDrawTool) { delete m_pDrawTool; m_pDrawTool = nullptr; } FreeLibrary(hModule);
通过这种方式,CCVideoWnd 类可以调用 DLL 中的 CDrawTool,而无需直接将 CDrawTool 放在主程序里。
相关文章:
创建插件 DLL 项目
Step 1: 创建插件 DLL 项目 在 Visual Studio 中创建一个新的 DLL 项目,并添加以下文件和代码。 头文件:CShapeBase.h cpp 复制代码 #pragma once #include <afxwin.h> // MFC 必需头文件 #include <string> #include <vector> #i…...
OpenCV双目相机外参标定C++
基于OpenCV库实现双目测量系统外参标定过程。通过分析双目测量系统左右相机拍摄的棋盘格标定板图像,包括角点检测、立体标定、立体校正和畸变校正的步骤,获取左右相机的相对位置关系和姿态。 a.检测每张图像中的棋盘格角点,并进行亚像素级精…...
【GESP】C++一级练习BCQM3055,4位数间隔输出
一级知识点取余、整除运算和格式化输出知识点应用。其实也可以用string去处理,那就属于GESP三级的知识点范畴了,孩子暂未涉及。 题目题解详见:https://www.coderli.com/gesp-1-bcqm3055/ https://www.coderli.com/gesp-1-bcqm3055/https://w…...
纯血鸿蒙的最难时刻才开始
关注卢松松,会经常给你分享一些我的经验和观点。 纯血鸿蒙(HarmonyOS NEXT)也正式发布了,绝对是一个历史性时刻,但最难的鸿蒙第二个阶段,也就是生态圈的建设,才刚刚开始。 目前,我劝你现在不要升级到鸿蒙…...
记一个mysql的坑
数据库表user, 存在一个name字段,字段为varchar类型 现在user表有这么两条记录: idnameageclass1NULL18一班2lisi20二班 假如我根据下面这一条件去更新,更新成功数据行显示为0 update user set age 19 where age 18 and class “一班”…...
Java中的设计模式:单例模式详解
摘要 单例模式(Singleton Pattern)是Java中最常用的设计模式之一,属于创建型模式。它的主要目的是确保一个类在系统中只有一个实例,并提供一个全局访问点来访问该实例。 1. 单例模式的定义 单例模式确保一个类只有一个实例&…...
NanoTrack原理与转tensorrt推理
文章目录 前言一、NanoTrack 工作原理二、运行demo与转换tensorrt模型2.1 运行pt模型demo2.2 转onnx模型2.3 转tensorrt模型2.4 运行trt模型推理 三、推理速度对比总结 前言 NanoTrack 是一种轻量级且高效的目标跟踪算法,基于Siamese网络架构,旨在在资源…...
YOLO11改进 | 卷积模块 | 卷积模块替换为选择性内核SKConv【附完整代码一键运行】
秋招面试专栏推荐 :深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转 💡💡💡本专栏所有程序均经过测试,可成功执行💡💡💡 本文给大家带来的教程是将YOLO11的卷积替…...
CentOS进入单用户模式进行密码重置
一、单用户模式介绍 单用户模式是一种特殊的启动模式,主要用于系统维护和故障排除。在单用户模式下,系统以最小化的状态启动,只有最基本的系统服务会被加载,通常只有root用户可以登录。这种模式提供了对系统的完全控制࿰…...
bitpoke- mysql-operator cluster
sidecar版本只支持到8.0.35,35可以支持到mysql8.0.35 . 默认镜像是5.7的。需要自己打sidecar的镜像: # Docker image for sidecar containers # https://github.com/bitpoke/mysql-operator/tree/master/images/mysql-operator-sidecar-8.0 # 参考5…...
第5课 基本数据类型
一、数据类型的诞生 在Python的世界里,万物皆对象,每个对象都有自己的若干属性,每一个属性都能描述对象的某一个方面。就像我们每个人,都有自己的身高、年龄、姓名、性别等很多方面的信息,这里的身高、年龄、姓名、性…...
OceanBase 首席科学家阳振坤:大模型时代的数据库思考
2024年 OceanBase 年度大会 即将于10月23日,在北京举行。 欢迎到现场了解更多“SQL AI ” 的探讨与分享! 近期,2024年金融业数据库技术大会在北京圆满举行,聚焦“大模型时代下数据库的创新发展”议题,汇聚了国内外众多…...
国内知名的几个镜像源
在国内,有许多常用的Python库镜像源可以帮助加速库的下载。以下是几个知名的镜像源: 1. 清华大学TUNA协会 网址: https://pypi.tuna.tsinghua.edu.cn/simple命令示例:pip install numpy --index-url https://pypi.tuna.tsinghua.edu.cn/simple2. 阿里云…...
海外著名新闻门户媒体软文发稿之华盛顿独立报-大舍传媒
在当今全球化的时代,信息传播的速度和范围达到了前所未有的程度。对于企业和个人而言,如何在国际舞台上有效地展示自己、传递信息,成为了一项至关重要的任务。而海外媒体发稿,特别是通过像华盛顿独立报这样的知名新闻门户…...
青少年编程与数学 02-002 Sql Server 数据库应用 13课题、函数的编写
青少年编程与数学 02-002 Sql Server 数据库应用 13课题、函数的编写 课题摘要:一、函数内置函数用户定义的函数 (User-Defined Functions, UDFs)使用示例主要特点 二、内置函数数学函数(Mathematical Functions)字符串函数(String Functions…...
关于LaTeX的floatrow包导入后标题无法直接放到浮动体上方
排版一个文章,标题怎么弄都弄不到表格上方,经过阅读帮助文档才发现问题。又是一个坑。 标题位置控制 使用floatrow包之后,类似 \begin{table}[htbp]\caption{xxx。}\label{table1}\centering\begin{tabular}{lcccc}\toprule& \multicol…...
Flutter Image和Text图文组件实战案例
In this section, we’ll go through the process of building a user interface that showcases a product using the Text and Image widgets. We’ll follow Flutter’s best practices to ensure a clean and effective UI structure. 在本节中,我们将使用“Te…...
使用 xlrd 和 xlwt 库进行 Excel 文件操作
使用 xlrd 和 xlwt 库进行 Excel 文件操作 在数据分析和处理的过程中,Excel 文件是最常用的数据存储格式之一。Python 提供了多种库来处理 Excel 文件,其中 xlrd 和 xlwt 是两个经典的库,分别用于读取和写入 Excel 文件。本文将详细介绍如何使用这两个库进行 Excel 文件的操…...
03.04、化栈为队
03.04、化栈为队 1、题目描述 实现一个 MyQueue 类,该类用两个栈来实现一个队列。 2、解题思路 本题要求使用两个栈来实现一个队列。队列遵循先进先出(FIFO)的原则,而栈遵循后进先出(LIFO)的原则。因此…...
Coppelia Sim (v-REP)仿真 机器人3D相机手眼标定与实时视觉追踪 (二)
coppelia sim[V-REP]仿真实现 机器人于3D相机手眼标定与实时视觉追踪 二 zmq API接口python调用python获取3D相机的数据获取彩色相机的数据获取深度相机的数据用matpolit显示 python控制机器人运动直接控制轴的位置用IK运动学直接移动到末端姿态 相机内参的标定记录拍照点的位置…...
深入AD9361:除了QPSK和FM,这颗射频芯片在Zynq平台上还能玩出什么花样?
深入AD9361:解锁Zynq平台上的射频创新潜能 当工程师们首次接触AD9361这颗射频芯片时,往往会被其标准应用场景如QPSK调制或FM收音所吸引。然而,这颗高度集成的RF收发器IC的真正价值,在于它为Zynq PSPL架构带来的无限可能性。本文将…...
矿用防爆监控哪家更值得选择
导读:矿用防爆监控是保障煤矿等高危工业场景安全的重要设备,其性能和可靠性直接关系到生产安全。在众多品牌中,如何选择一款优质的矿用防爆监控产品成为了许多企业的关注焦点。本文将从多个维度对矿用防爆监控进行深度解析,并推荐…...
Cursor Pro功能完全解锁指南:三步实现免费无限使用终极方案
Cursor Pro功能完全解锁指南:三步实现免费无限使用终极方案 【免费下载链接】cursor-free-vip [Support 0.45](Multi Language 多语言)自动注册 Cursor Ai ,自动重置机器ID , 免费升级使用Pro 功能: Youve reached you…...
程序员转大模型,这8个必备框架,新手也能快速上手
文章目录前言一、为什么2026年必须学大模型开发?二、8个必备框架详细解析2.1 LangChain:AI应用开发的"事实标准"核心优势2026年最新变化适用场景新手快速上手避坑指南2.2 LangGraph:企业级智能体的"骨架"核心优势为什么2…...
GanttProject项目管理软件:完全免费的甘特图工具使用指南
GanttProject项目管理软件:完全免费的甘特图工具使用指南 【免费下载链接】ganttproject Official GanttProject repository. 项目地址: https://gitcode.com/gh_mirrors/ga/ganttproject GanttProject是一款功能强大的免费开源项目管理软件,专为…...
基于大语言模型的智能BI工具:从自然语言到SQL与可视化的工程实践
1. 项目概述:一个开源的商业智能对话工具最近在折腾数据分析和可视化,发现一个挺有意思的开源项目,叫openchatbi。简单来说,它就是一个能让你用自然语言跟数据库“聊天”的工具。你不需要写复杂的 SQL 语句,直接问“上…...
Claude Code 模型特定调优与 A/B 测试全解析:Feature Flag、灰度发布、Undercover、安全门控、Prompt 调优与 AI Agent 工程化实战
一、先说结论:AI Agent 真正难的不是“会调用模型”,而是“能持续驾驭模型”很多人做 AI 编码助手、企业智能体、研发提效工具时,第一反应是接入一个更强的大模型:换成更大的参数、更新的版本、更长的上下文,似乎问题就…...
Adobe GenP 3.0终极指南:3步解锁全系列Adobe CC软件
Adobe GenP 3.0终极指南:3步解锁全系列Adobe CC软件 【免费下载链接】Adobe-GenP Adobe CC 2019/2020/2021/2022/2023 GenP Universal Patch 3.0 项目地址: https://gitcode.com/gh_mirrors/ad/Adobe-GenP 还在为Adobe Creative Cloud高昂的订阅费用而烦恼吗…...
基于RAG与德国开放数据构建本地化智能问答系统实践
1. 项目概述与核心价值最近在折腾本地化大语言模型应用时,发现了一个挺有意思的项目:stefangrotz/OpenDataGermanyGPT。光看名字,你可能会觉得这又是一个针对特定地区数据的聊天机器人,没什么新意。但实际深入进去,你会…...
provision-cli:构建组织级基础设施即代码标准化工作流
1. 项目概述:一个为组织级基础设施管理而生的命令行工具如果你在管理一个稍具规模的技术团队,或者负责一个拥有多个项目、环境(开发、测试、生产)的软件产品,那么你一定对“基础设施即代码”这个概念不陌生。但当你真正…...
