[连载]《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 ...
随机推荐
- iOS10之Expected App Behaviors
昨天上架到appStore的时候碰到个问题,构建好后上传到itunesconnect的的包都用不了, 显示错误为:此构建版本无效. 或者英文显示为:ITC.apps.preReleaseBuild.e ...
- jQuery.data() 使用方法
data() 方法向被选元素附加数据,或者从被选元素获取数据.在实际开发中,可以用来记录上一步操作某一对象的值,来给下一步操作做一些判断 $("#btn1").click(func ...
- emmet,jade,haml, slim,less,sass,coffeescript等的实战优缺点
摘要: 文章背景,来自于群内周五晚上的一次头脑风暴式的思维碰撞交流活动. 随着前端技术的蓬勃发展, 各种新技术随着生产力的需要不断的涌入我们的视野, 那今天探讨的话题是这些新时代的前端兵器谱: 一. ...
- 【腾讯Bugly干货分享】Android进程保活招式大全
本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57ac4a0ea374c75371c08ce8 作者:腾讯——张兴华 目前市面上 ...
- Atitit 数据存储视图的最佳实际best practice attilax总结
Atitit 数据存储视图的最佳实际best practice attilax总结 1.1. 视图优点:可读性的提升1 1.2. 结论 本着可读性优先于性能的原则,面向人类编程优先于面向机器编程,应 ...
- webSocket and LKDBHelper的使用说明
socketket与lkdbhelper来处理数据 客户需求: 当我们有需要从自己的后台推送消息给我们的用户时,用户需要实时的接收到来自我们的推送消息.前提是没有使用第三方的推送框架,那么这个使用we ...
- Android点滴
1,View中getWidth(),getLayoutParams.width,getMeasureedWidth()的区别 2,setCompoundDrawables和setCompoundDra ...
- sublime text 插件
html-css-js prettify html5 anaconda pretty json JavaScript Completions SqlBeautifier Emmet Css Snipp ...
- javascript学习之BOM
BOM是browser object model的缩写,简称浏览器对象模型.先看看下面这张图 window对象是BOM的顶层(核心)对象,所有对象都是通过它延伸出来的,也可以称为window的子对象. ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(21)-权限管理系统-跑通整个系统
系列目录 这一节我们来跑通整个系统,验证的流程,通过AOP切入方式,在访问方法之前,执行一个验证机制来判断是否有操作权限(如:增删改等) 原理:通过MVC自带筛选器,在筛选器分解路由的Action和c ...