[连载]《C#通讯(串口和网络)框架的设计与实现》- 8.总体控制器的设计
目 录
第八章 总体控制器的设计... 2
8.1 总控制器的职能... 2
8.2 组装和释放部件... 3
8.3 事件响应... 5
8.4 小结... 9
第八章 总体控制器的设计
有了IO部分、设备驱动部分、显示部分、数据导出部分和服务组件部分等,在这些已经存在的接口上构建一个集成各部分的总控制器,协调各部分有序工作、事件响应和控制数据流向。
另外,这个总控制器还负责与宿主程序进行交互,可以这样理解:总控制器是宿主程序与IO部分、设备驱动部分、显示部分、数据导出部分和服务组件部分之间交互的载体,并且是唯一的。结构示意图如下:

8.1 总控制器的职能
总控制器(IDeviceController)的职能包括:增加和删除设备驱动、增加导出数据实例、增加图形显示实例、增加服务组件实例、单击服务事件、释放控制器资源等等。接口定义如下图:

8.2 组装和释放部件
DeviceController是总控制器的实例体类,继承自IDeviceController接口。通过构造函数来完成对总控制器的初始化,代码如下:
public DeviceController()
{
_devList = DeviceManager.GetInstance();
_ioController = IOControllerManager.GetInstance();
_runContainer = RunContainerForm.GetRunContainer();
_runContainer.MouseRightContextMenuHandler += RunContainer_MouseRightContextMenuHandler;
_dataShowController = new GraphicsShowController();
_exportController = new ExportDataController();
_appServiceManager = new AppServiceManager();
}
通过ReleaseDeviceController接口来完成对总控制器的资源释放,代码如下:
public void ReleaseDeviceController()
{
_ioController.RemoveAllController();
_runContainer.RemoveAllDevice();
_runContainer.MouseRightContextMenuHandler -= RunContainer_MouseRightContextMenuHandler;
_exportController.RemoveAll();
_dataShowController.RemoveAll();
_appServiceManager.RemoveAll();
IEnumerator<IRunDevice> devList = _devList.GetEnumerator();
while (devList.MoveNext())
{
devList.Current.ExitDevice();
}
_devList.RemoveAllDevice();
}
软件退出时释放资源要比软件启动时加载资源要复杂的多,这块涉及到两方面问题:(1)释放资源顺序,如果资源提前释放,那么往往会造成后边代码在执行过程中出现无法引用对象资源的现象,造成意想不到的结果,所以一定要对实例的可用性进行判断。(2)事务性的线程无法正常退出,造成软件界面已经关闭,但是后台进程却一直存在。特别是对线程退出的处理,框架平台采用了统一的线程退出机制,代码如下:
public void StartThead()
{
if (_RunThread == null || !_RunThread.IsAlive)
{
this._IsExit = false;
this._RunThread = new Thread(new ThreadStart(RunThead));
this._RunThread.IsBackground = true; //该线程为后台线程
this._RunThread.Name = "RunThread";
this._RunThread.Start();
}
} private void RunThead()
{
while (!_IsExit)
{
if(_IsExit) //如果标识为true,则退出循环,退出线程
{
break;
}
//事务处理
}
} public void StopThead()
{
if (this._RunThread != null && this._RunThread.IsAlive)
{
this._IsExit = true; //标识当前线程为可退出线程。
this._RunThread.Join(1000);//阻塞调用线程,直到某个线程终止或经过了指定时间为止
try
{
_RunThread.Abort(); //为了防止线程没有退出,进行强行终止,有可能造成文件损坏
}
catch
{
}
}
}
8.3 事件响应
增加和删除设备驱动都会对设备的事件进行绑定和解绑。代码如下:
dev.DeviceRuningLogHandler += new DeviceRuningLogHandler(DeviceRuningLogHandler);
dev.UpdateContainerHandler += new UpdateContainerHandler(UpdateContainerHandler);
dev.DeviceObjectChangedHandler += new DeviceObjectChangedHandler(DeviceObjectChangedHandler);
dev.ReceiveDataHandler += new ReceiveDataHandler(ReceiveDataHandler);
dev.SendDataHandler += new SendDataHandler(SendDataHandler);
dev.COMParameterExchangeHandler += new COMParameterExchangeHandler(COMParameterExchangeHandler);
dev.DeleteDeviceHandler += new DeleteDeviceHandler(DeleteDeviceHandler);
具体含义请参见《第3章 设备驱动的设计》中的“3.12 事件响应设计”,COMParameterExchangeHandler改变串口参数事件响应代码如下:
private void COMParameterExchangeHandler(object source, COMParameterExchangeArgs e)
{
if (e == null)
{
return;
}
IRunDevice dev = this._devList.GetDevice(e.DeviceID.ToString());
if (dev != null)
{
if (dev.CommunicationType == CommunicationType.COM)
{
if (e.OldCOM != e.NewCOM)
{
//--------------对旧串口进行处理----------------//
IRunDevice[] oldCOMDevList = this._devList.GetDevices(e.OldCOM.ToString(), CommunicationType.COM);
//---------------检测当前串口设备数------------//
int existCOMCount = 0;
for (int i = 0; i < oldCOMDevList.Length; i++)
{
if (oldCOMDevList[i].GetHashCode() != dev.GetHashCode())
{
existCOMCount++;
}
}
//------------------------------------------//
if (existCOMCount <= 0)//该串口没有可用的设备
{
IIOController oldCOMController = IOControllerManager.GetInstance().GetController(SessionCom.FormatKey(e.OldCOM));
if (oldCOMController != null)
{
_ioController.CloseController(oldCOMController.Key);
} else
{
DeviceMonitorLog.WriteLog(e.DeviceName, "该设备的串口控制器为空");
}
}
//--------------对新串口进行处理----------------//
bool newCOMControllerExist = IOControllerManager.GetInstance().ContainController(SessionCom.FormatKey(e.NewCOM));
if (!newCOMControllerExist)
{
IIOController newCOMController = _ioController.BuildController(e.NewCOM.ToString(), e.NewBaud.ToString(), CommunicationType.COM);
if (newCOMController != null)
{
newCOMController.StartService();
_ioController.AddController(newCOMController.Key.ToString(), newCOMController);
}
else
{
DeviceMonitorLog.WriteLog(e.DeviceName, "创建该设备的串口控制器失败");
}
}
DeviceMonitorLog.WriteLog(e.DeviceName, String.Format("串口从{0}改为{1}", e.OldCOM.ToString(), e.NewCOM.ToString())); }
else
{
if (e.OldBaud != e.NewBaud)
{
ISessionCom comIO = (ISessionCom)SessionComManager.GetInstance().GetIO(SessionCom.FormatKey(e.OldCOM));
if (comIO != null)
{
bool success = comIO.IOSettings(e.NewBaud);
if (success)
{
DeviceMonitorLog.WriteLog(e.DeviceName, String.Format("串口{0}的波特率从{1}改为{2}成功", e.OldCOM.ToString(), e.OldBaud.ToString(), e.NewBaud.ToString()));
}
else
{
DeviceMonitorLog.WriteLog(e.DeviceName, String.Format("串口 {0} 的波特率从 {1} 改为 {2} 失败", e.OldCOM.ToString(), e.OldBaud.ToString(), e.NewBaud.ToString()));
}
}
}
}
}
else
{
DeviceMonitorLog.WriteLog(e.DeviceName, "不是串口类型的设备");
}
}
}
同时,还包括GraphicsShowClosedHandler和MouseRightContextMenuHandler两个事件。当关闭显示视图的时候会触发GraphicsShowClosedHandler事件,把当前视图从管理器中移除,并释放资源;当右键单击显示视图会触发MouseRightContextMenuHandler事件,以调用相应设备的上下文菜单。
8.4 小结
总体控制器不是必须的,宿主程序完全可以直接与IO部分、设备驱动部分、显示部分、数据导出部分和服务组件部分进行交互。但是,为了结构清晰、方便扩展,在中间加了一层进行总体协调。
作者:唯笑志在
Email:504547114@qq.com
QQ:504547114
.NET开发技术联盟:54256083
文档下载:http://pan.baidu.com/s/1pJ7lZWf
官方网址:http://www.bmpj.net
[连载]《C#通讯(串口和网络)框架的设计与实现》- 8.总体控制器的设计的更多相关文章
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 0.前言
目 录 前言 前言 刚参加工作,使用过VB.VC开发软件,随着C#的崛起,听说是C++++,公司决定以后开发软件使用C#,凭借在 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-3.设备驱动的设计
目 录 第三章 设备驱动的设计... 2 3.1 初始化设备... 4 3.2 运行设备接口设计... 4 3.3 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 10.宿主程序详细设计
目 录 第十章 宿主程序详细设计... 2 10.1 配置文件设计... 3 10.2 加载设备驱动... 4 10.3 加载 ...
- 《连载 | 物联网框架ServerSuperIO教程》-4.如开发一套设备驱动,同时支持串口和网络通讯。附:将来支持Windows 10 IOT
1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制. <连载 | 物联网框架Serve ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-1.通讯框架介绍
[连载]<C#通讯(串口和网络)框架的设计与实现>- 0.前言 目 录 第一章 通讯框架介绍... 2 1.1 通讯的本质... 2 1 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 6.通讯控制器的设计
目 录 第六章 通讯控制器的设计... 2 6.1 控制器接口... 2 6.2 串口控制器... 3 6.3 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-2.框架的总体设计
目 录 C#通讯(串口和网络)框架的设计与实现... 1 (SuperIO)- 框架的总体设计... 1 第二章 框架总体的设计... 2 2.1 ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》- 5.串口和网络统一IO设计
目 录 第五章 串口和网络统一IO设计... 2 5.1 统一IO接口... 2 5.1.1 串口IO.. 4 5.1.2 网络IO.. ...
- [连载]《C#通讯(串口和网络)框架的设计与实现》-4.设备驱动管理器的设计
目 录 第四章 设备驱动管理器的设计... 2 4.1 接口定义... 2 4.2 设备容器... 7 4.3 ...
随机推荐
- 熊乐:H3 BPM为加速企业流程管理提供源动力
近日,在北京·金隅喜来登酒店,H3 BPM以"让天下没有难用的流程"为主题,正式发布H3 BPM10.0版本.全新的业务流程管理系统在易用性方面大大提升,并且全面支持Java与.N ...
- BPM配置故事之案例1-配置简单流程
某天,Boss找到了信息部工程师小明. Boss:咱们新上了H3 BPM,你研究研究把现在的采购申请流程加上去吧,这是采购申请单. 小明:好嘞 采购申请单 小明回去后拿着表单想了想,开始着手配置. 他 ...
- Mach-O 的动态链接(Lazy Bind 机制)
➠更多技术干货请戳:听云博客 动态链接 要解决空间浪费和更新困难这两个问题最简单的方法就是把程序的模块相互分割开来,形成独立的文件,而不再将它们静态的链接在一起.简单地讲,就是不对那些组成程序的目标文 ...
- Struts2框架基础
Struts2框架基础 1.Java的框架 1.1.框架简介 在大型项目开发过程中,经常会使用到一些框架,这样做好的好处是能够提高工作效率,在java中最常用的的框架就是SSH,这其实是三个框架的简称 ...
- 将MPM雪模拟移植到Maya
同事实现了一个迪士尼的MPM雪模拟论文,我将其移植到Maya中 论文题目是 A material point method for snow simulation 代码在这里: https://git ...
- Johnson 全源最短路径算法
解决单源最短路径问题(Single Source Shortest Paths Problem)的算法包括: Dijkstra 单源最短路径算法:时间复杂度为 O(E + VlogV),要求权值非负: ...
- MongoDB与PostgresQL无责任初步测试
PostgresQL一秒能插入多少条记录,MongoDB呢?读取的情况又如何?我写了一些简单的程序,得出了一些简单的数据,贴在这里分享,继续往下阅读前请注意下本文标题中的“无责任”,这表示此测试结果不 ...
- C++的性能C#的产能?! - .Net Native 系列四:性能测试方法(PerfView)
之前一文<c++的性能, c#的产能?!鱼和熊掌可以兼得,.NET NATIVE初窥> 获得很多朋友支持和鼓励,也更让我坚定做这项技术的推广者,希望能让更多的朋友了解这项技术,于是先从官方 ...
- some OpenGL constants
some OpenGL constants This is from (https://github.com/peterderivaz/pyopengles/blob/master/gl2.py) G ...
- C#设计模式-责任链模式
在现实生活中,有很多请求并不是一个人说了就算的,例如面试时的工资,低于1万的薪水可能技术经理就可以决定了,但是1万~1万5的薪水可能技术经理就没这个权利批准,可能就需要请求技术总监的批准,所以在面试的 ...