一种松耦合的分层插件系统的设计和实现
分享一下我老师大神的人工智能教程!零基础,通俗易懂!http://blog.csdn.net/jiangjunshow
也欢迎大家转载本篇文章。分享知识,造福人民,实现我们中华民族伟大复兴!
作者:朱金灿
来源:http://blog.csdn.net/clever101
C++编写的桌面软件中传统的App+Dll的架构已经显现出的模块之间的强耦合、维护性差、升级不方便等诸多弊端,为此我进行了一些思考,有兴趣的朋友请看我以前的一个思考片断:《Windows平台下C++插件系统实现的几个关键技术问题及其解决思路》,今天则具体介绍的我的一个具体设计和实现。
我的目标是要去除功能模块的相互依赖,在模块调用必须采用动态加载的办法,但同时各个模块可以进行自由地进行通讯。我的设想大致是这样的:在动态加载各个模块后调用统一接口后生成插件对象,在某个插件对象都能通过标识符找到其它的插件对象,通过统一的接口将数据传给它们。系统架构图如下:
系统流程图如下:
大家可以看到,要构建这样的系统需要解决两个关键问题:
1. 如何动态加载dll创建插件对象以及插件基类对象的接口设计
2. 插件模块之间的通讯问题
如何解决这两个问题我已有了基本的思路。为了验证我的思路是可行的,我用VC2008新建了一个LayeredArchit解决方案,该解决方案实现这样一个简单功能:统计指定目录下的文件数,然后将统计结果保存到一个文本文件。这个解决方案由以下工程组成:
BaseObjLib —— 底层库,用于定义底层对象和接口
StatFile —— 统计文件插件
OutputFile —— 将统计结果输出到文本文件的插件
CmdApp —— 一个主调用程序(是一个控制台程序)。
在每个插件模块都实现一个插件对象,继承基类插件对象IPluginObj,里都有这样一个统一的插件导出接口来生成插件对象:
extern "C" __declspec( dllexport ) IPluginObj* __stdcall GetPluginObj(const std::string &strID)
在动态加载插件库之后就调用这个接口。这样就解决了动态创建插件对象的问题。
接下来我们解决插件之间的数据通讯问题。前面的设计已经实现了在任何插件模块里都能找到其它的插件对象。一个很自然的设计是在基类定义这样一个接口:
virtual BOOL ProcessData(void* pData){ return FALSE;}
然后在派生插件对象里重载这个接口。这个设计基本可以实现数据通讯,但是有一个严重弊端是void*是类型不安全的,假如数据接收方将其类型转换错了程序就很可能崩溃。特别是当一个插件需要接受两种类型数据时。因此必须解决类型安全问题。
之后我想到了利用微软的com模型的万能类型VARIANT,但是看了一篇文章《用户自定义结构数据与VARIANT转换》就放弃了,感觉转换太繁琐了。后面惊喜地发现原来boost库早已实现了一个万能类型boost::any。boost::any真的很符合我的需求。具体是在基类插件对象类定义一个虚接口:
virtual BOOL ProcessData(boost::any& anyData){ return FALSE;}
在主程序里找到统计插件对象,并将路径传递给它:
IPluginObjPtr ptrStatFile = pPluginFactory->GetPlugin(_T("StatFile.dll")); std::string strFolder = _T("D:\\dev-utility-tools\\doc"); boost::any anyString = strFolder; ptrStatFile->ProcessData(anyString);
// 统计模块的ProcessData函数的实现
BOOL CStatFilePlugin::ProcessData( boost::any& anyData ){ if(anyData.type() == typeid(std::string)) { std::string strPath = boost::any_cast<std::string>(anyData); if(::GetFileAttributes(strPath.c_str())==-1) { std::cout<<"目录不存在"<<std::endl; return FALSE; } m_info.m_strPath = strPath; std::cout<<"现在正在统计的是:"<<m_info.m_strPath<<std::endl; std::cout<<"请等待!!!"<<std::endl; std::string strTmpDir = m_info.m_strPath + TEXT("\\"); // 统计文件数 StatAllFileInFolder(strTmpDir); std::cout<<"文件夹数:"<<m_info.m_FolderNum<<std::endl; std::cout<<"文件数:"<<m_info.m_FileNum<<std::endl;// 找到输出到文件插件对象,将统计结果传给它 CPluginFactory* pPluginFactory = CPluginFactory::Instance(); IPluginObjPtr ptrOuttoFile = pPluginFactory->GetPlugin(_T("OutputFile.dll")); boost::any anyInfo = m_info; ptrOuttoFile->ProcessData(anyInfo); return TRUE; } return FALSE;}
这样就安全地实现了插件间的数据通讯。需要指出的是需要通讯的插件间不相互依赖,但需要都包含的定义了传递数据结构体的头文件。具体请见源码下的DataExchange文件夹,该文件夹就是定义需要通讯的数据结构体。
程序的效果图如下:
一个扩展设想是:消息本质上也是一种数据。消息的传递也可以参考数据的传递来实现。
源码已上传至http://code.google.com/p/dev-utility-tools/source(具体见PluginArchit解决方案),采用GPL V2.0开源协议,有兴趣请下载看看。
参考文献:
1. 4.6.1 万能类型boost::any
给我老师的人工智能教程打call!http://blog.csdn.net/jiangjunshow
你好! 这是你第一次使用 **Markdown编辑器** 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章,了解一下Markdown的基本语法知识。
新的改变
我们对Markdown编辑器进行了一些功能拓展与语法支持,除了标准的Markdown编辑器功能,我们增加了如下几点新功能,帮助你用它写博客:
- 全新的界面设计 ,将会带来全新的写作体验;
- 在创作中心设置你喜爱的代码高亮样式,Markdown 将代码片显示选择的高亮样式 进行展示;
- 增加了 图片拖拽 功能,你可以将本地的图片直接拖拽到编辑区域直接展示;
- 全新的 KaTeX数学公式 语法;
- 增加了支持甘特图的mermaid语法1 功能;
- 增加了 多屏幕编辑 Markdown文章功能;
- 增加了 焦点写作模式、预览模式、简洁写作模式、左右区域同步滚轮设置 等功能,功能按钮位于编辑区域与预览区域中间;
- 增加了 检查列表 功能。
功能快捷键
撤销:Ctrl/Command + Z
重做:Ctrl/Command + Y
加粗:Ctrl/Command + B
斜体:Ctrl/Command + I
标题:Ctrl/Command + Shift + H
无序列表:Ctrl/Command + Shift + U
有序列表:Ctrl/Command + Shift + O
检查列表:Ctrl/Command + Shift + C
插入代码:Ctrl/Command + Shift + K
插入链接:Ctrl/Command + Shift + L
插入图片:Ctrl/Command + Shift + G
合理的创建标题,有助于目录的生成
直接输入1次#,并按下space后,将生成1级标题。
输入2次#,并按下space后,将生成2级标题。
以此类推,我们支持6级标题。有助于使用TOC
语法后生成一个完美的目录。
如何改变文本的样式
强调文本 强调文本
加粗文本 加粗文本
标记文本
删除文本
引用文本
H2O is是液体。
210 运算结果是 1024.
插入链接与图片
链接: link.
图片:
带尺寸的图片:
当然,我们为了让用户更加便捷,我们增加了图片拖拽功能。
如何插入一段漂亮的代码片
去博客设置页面,选择一款你喜欢的代码片高亮样式,下面展示同样高亮的 代码片
.
// An highlighted block var foo = 'bar';
生成一个适合你的列表
- 项目
- 项目
- 项目
- 项目
- 项目1
- 项目2
- 项目3
- 计划任务
- 完成任务
创建一个表格
一个简单的表格是这么创建的:
项目 | Value |
---|---|
电脑 | $1600 |
手机 | $12 |
导管 | $1 |
设定内容居中、居左、居右
使用:---------:
居中
使用:----------
居左
使用----------:
居右
第一列 | 第二列 | 第三列 |
---|---|---|
第一列文本居中 | 第二列文本居右 | 第三列文本居左 |
SmartyPants
SmartyPants将ASCII标点字符转换为“智能”印刷标点HTML实体。例如:
TYPE | ASCII | HTML |
---|---|---|
Single backticks |
'Isn't this fun?'
|
‘Isn’t this fun?’ |
Quotes |
"Isn't this fun?"
|
“Isn’t this fun?” |
Dashes |
-- is en-dash, --- is em-dash
|
– is en-dash, — is em-dash |
创建一个自定义列表
- Markdown
- Text-to- HTML conversion tool
- Authors
- John
- Luke
如何创建一个注脚
一个具有注脚的文本。2
注释也是必不可少的
Markdown将文本转换为 HTML。
KaTeX数学公式
您可以使用渲染LaTeX数学表达式 KaTeX:
Gamma公式展示 Γ ( n ) = ( n − 1 ) ! ∀ n ∈ N \Gamma(n) = (n-1)!\quad\forall n\in\mathbb N Γ(n)=(n−1)!∀n∈N 是通过欧拉积分
Γ ( z ) = ∫ 0 ∞ t z − 1 e − t d t   . \Gamma(z) = \int_0^\infty t^{z-1}e^{-t}dt\,. Γ(z)=∫0∞tz−1e−tdt.
你可以找到更多关于的信息 LaTeX 数学表达式here.
新的甘特图功能,丰富你的文章
ganttdateFormat YYYY-MM-DDtitle Adding GANTT diagram functionality to mermaidsection 现有任务已完成 :done, des1, 2014-01-06,2014-01-08进行中 :active, des2, 2014-01-09, 3d计划一 : des3, after des2, 5d计划二 : des4, after des3, 5d
- 关于 甘特图 语法,参考 这儿,
UML 图表
可以使用UML图表进行渲染。 Mermaid. 例如下面产生的一个序列图::
这将产生一个流程图。:
- 关于 Mermaid 语法,参考 这儿,
FLowchart流程图
我们依旧会支持flowchart的流程图:
- 关于 Flowchart流程图 语法,参考 这儿.
导出与导入
导出
如果你想尝试使用此编辑器, 你可以在此篇文章任意编辑。当你完成了一篇文章的写作, 在上方工具栏找到 文章导出 ,生成一个.md文件或者.html文件进行本地保存。
导入
如果你想加载一篇你写过的.md文件或者.html文件,在上方工具栏可以选择导入功能进行对应扩展名的文件导入,
继续你的创作。
mermaid语法说明 ↩︎
注脚的解释 ↩︎
一种松耦合的分层插件系统的设计和实现相关推荐
- 哈工大视听觉信号处理——听觉部分报告——一种智能家居命令词识别系统的设计
题 目 听觉部分 专 业 软件工程 学 号 120L0219XX 班 级 2037101 姓 名 李启明 报 告 日 期 2022.12.20 一.基频 ( ...
- 一种基于星型结构的系统架构设计
随着企业发展,传统的中心化,大平台的软件系统设计越来越难以满足现实要求.对于位于决策中心的集团总公司来说,制定管理规则,收集核心数据当然是IT系统的主要功能.对于一线业务部门来说,他们也需要在总公司所 ...
- 专访关东升:松耦合分层架构设计
关东升,国内知名iOS技术作家,iOS技术顾问,高级培训讲师,移动开发专家.拥有16年软件开发经验.8年培训行业经验.精通iOS.Android和 Windows Phone 7及Html5等移动开发 ...
- 松耦合和紧耦合的架构设计、性能对比
在最近的一次大数据技术讨论会上,有一家公司的技术高管谈到松耦合和紧耦合的性能表现的话题.正好Laxcus大数据管理系统的设计,从0.x.1.x到2.x版本,也经历了从紧耦合到松耦合的发展过程.做为亲历 ...
- 迪米特法则实现“高内聚、松耦合”
什么是"高内聚.松耦合"? 如何利用迪米特法则来实现"高内聚.松耦合"? 有哪些代码设计是明显违背迪米特法则的?对此又该如何重构? 何为"高内聚.松耦 ...
- 【设计模式之美 设计原则与思想:设计原则】22 | 理论八:如何用迪米特法则(LOD)实现“高内聚、松耦合”?
今天,我们讲最后一个设计原则:迪米特法则.尽管它不像 SOLID.KISS.DRY 原则那样,人尽皆知,但它却非常实用.利用这个原则,能够帮我们实现代码的"高内聚.松耦合".今天, ...
- 如何用迪米特法则(LOD)实现“高内聚、松耦合”?
什么是"高内聚.松耦合"? 如何利用迪米特法则来实现"高内聚.松耦合"? 有哪些代码设计是明显违背迪米特法则的?对此又该如何重构? 何为"高内聚.松耦 ...
- 各种耦合关系,耦合,紧耦合,松耦合,解耦
松耦合和紧耦合的架构设计.性能对比 <div id="cnblogs_post_description" style="display: none"> ...
- INS/GNSS组合导航(三)松耦合、紧耦合、深度耦合
0 INS/GNSS组合导航概述 所谓INS/GNSS组合导航,就是利用INS和GNSS两者信息进行融合,从而综合利用两者的优点,实现优势互补,求解实现pose的解算,得到PVA(位置.速度.姿态)的 ...
最新文章
- cdialog创建后马上隐藏_都2020了,你还没用上这些Windows的“隐藏神技能”?
- 扫盲行动之九:Vi编辑器的基本使用方法!
- 【c语言 gcc9.1.0环境下编译报错】error: ‘true’ undeclared (first use in this function)
- base64编码_几分钟看懂Base64编码原理
- Python到底是什么?Python发展前景好吗?
- 关于Eclipse中复制粘贴一个项目后的操作
- 计算机网络 实验教案,《计算机网络》实验教案.pdf
- xp系统c 语言命令重定向,Xp命令解释程序cmd.exe概述
- Codeforces Round #462 (Div. 2): A. A Compatible Pair
- dependencies.dependency.version' for org.hibernate:hibernate-validator:jar is missing.
- 一张图了解大牛直播SDK 1
- 登录业务的演变、单点登录(SSO)的三种解决方案
- matlab提示利用了总线信号,Simulink 总线信号
- 快速使用easy rule
- 采用MCaaS模式 SAP为有孚网络提供基于HANA的解决方案及服务
- 判断点是否在点组成的封闭区域内c++
- itext操作word,设置页眉页脚,html转word
- h3c怎么设置虚拟服务器,h3c虚拟服务器设置
- 贪心算法|406. 根据身高重建队列|先排序后插队
- 02.Win10诸多坑之windows search服务启动失败
热门文章
- HDOJ(HDU) 2109 Fighting for HDU(简单排序比较)
- python打包分发工具:setuptools
- 怎么修改计算机用户登录密码是多少,怎么修改电脑密码?
- android 强制息屏,如何让Android设备实现息屏显示
- 微信公众号开发-vue1
- 【办公效率】更换笔记本屏幕后无法正常显示
- 《实例妙解 Cocos2d-x 游戏开发》反馈勘误
- sfc 修复 xp rpc服务器,我来告诉大家一些无耻的电脑知识(很牛逼很震撼,怕用的时候找不到...
- MyEclipse配置详细步骤(JDk版本1.7)
- Python+Win32 调用IE浏览器,却弹出360浏览器的解决方法