目       录

第八章           总体控制器的设计... 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.总体控制器的设计的更多相关文章

  1. [连载]《C#通讯(串口和网络)框架的设计与实现》- 0.前言

                                  目       录 前言 前言 刚参加工作,使用过VB.VC开发软件,随着C#的崛起,听说是C++++,公司决定以后开发软件使用C#,凭借在 ...

  2. [连载]《C#通讯(串口和网络)框架的设计与实现》-3.设备驱动的设计

    目       录 第三章           设备驱动的设计... 2 3.1           初始化设备... 4 3.2           运行设备接口设计... 4 3.3        ...

  3. [连载]《C#通讯(串口和网络)框架的设计与实现》- 10.宿主程序详细设计

    目       录 第十章           宿主程序详细设计... 2 10.1        配置文件设计... 3 10.2        加载设备驱动... 4 10.3        加载 ...

  4. 《连载 | 物联网框架ServerSuperIO教程》-4.如开发一套设备驱动,同时支持串口和网络通讯。附:将来支持Windows 10 IOT

    1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制. <连载 | 物联网框架Serve ...

  5. [连载]《C#通讯(串口和网络)框架的设计与实现》-1.通讯框架介绍

    [连载]<C#通讯(串口和网络)框架的设计与实现>- 0.前言 目       录 第一章           通讯框架介绍... 2 1.1           通讯的本质... 2 1 ...

  6. [连载]《C#通讯(串口和网络)框架的设计与实现》- 6.通讯控制器的设计

    目       录 第六章           通讯控制器的设计... 2 6.1           控制器接口... 2 6.2           串口控制器... 3 6.3          ...

  7. [连载]《C#通讯(串口和网络)框架的设计与实现》-2.框架的总体设计

    目       录 C#通讯(串口和网络)框架的设计与实现... 1 (SuperIO)- 框架的总体设计... 1 第二章           框架总体的设计... 2 2.1           ...

  8. [连载]《C#通讯(串口和网络)框架的设计与实现》- 5.串口和网络统一IO设计

    目       录 第五章           串口和网络统一IO设计... 2 5.1           统一IO接口... 2 5.1.1    串口IO.. 4 5.1.2    网络IO.. ...

  9. [连载]《C#通讯(串口和网络)框架的设计与实现》-4.设备驱动管理器的设计

    目       录 第四章           设备驱动管理器的设计... 2 4.1           接口定义... 2 4.2           设备容器... 7 4.3          ...

随机推荐

  1. JAVA环境变量和TomCat服务器配置

    Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP 程序的首选.对于一个初学者来说,可以这样 ...

  2. ios 类似微信红点显示功能

    设计思路:给UIView增加一个分类 所有的视图都可以根据需要来进行红点显示 #import <UIKit/UIKit.h> @interface UIView (CHRRedDot) @ ...

  3. 使用C# 和Consul进行分布式系统协调

    随着大数据时代的到来,分布式是解决大数据问题的一个主要手段,随着越来越多的分布式的服务,如何在分布式的系统中对这些服务做协调变成了一个很棘手的问题.今天我们就来看看如何使用C# ,利用开源对分布式服务 ...

  4. [异常解决] ubuntukylin16.04 LTS中关于flash安装和使用不了的问题解决

    http://www.linuxdiyf.com/linux/25211.html 归纳解决flash插件大法: 启动器中找到 软件更新,启动,点击 其它软件,把Canonical合作伙伴前方框 选上 ...

  5. seajs的使用

    写在前面 seajs是什么? Seajs是一个js文件加载器. 遵循 CMD 规范模块化开发,依赖的自动加载.配置的简洁清晰. 用于Web开发的模块加载工具,提供简单.极致的模块化体验 一:使用 文件 ...

  6. OpenCASCADE Job - Shoe Doctor

    鞋博士 鞋博士经过8年沉淀,在鞋类工业4.0全流程平台上积累了相当的技术实力,获投资商亲睐. 新的一年,在投资商协助下,将踏上新的征途,因此诚邀您加盟顶层技术合伙人. 如果您具备以下实力,我们期待您的 ...

  7. 【CSS进阶】伪元素的妙用2 - 多列均匀布局及title属性效果

    最近无论是工作还是自我学习提升都很忙,面对长篇大论的博文总是心有余而力不足,但又不断的接触学习到零碎的但是很有意义的知识点,很想分享给大家,所以本篇可能会很短. 本篇接我另一篇讲述 CSS 伪元素的文 ...

  8. clipChildren和clipToPadding

    clipChildren 和 clipToPadding clipChild 用于定义子控件是否在父控件边界内进行绘制.clipChild 默认为 true.也就是不允许进行扩展绘制. clipToP ...

  9. 浅析JavaScript事件流——冒泡

    一.什么是事件冒泡流 我们知道事件流指的是从页面中接受事件的顺序. 为了形象理解事件冒泡,可以想象三军主将诸葛亮,在帐内运筹帷幄,眼观六路耳听八方,这时候前方的战事情况就需要靠传令兵来传达,当第一位传 ...

  10. java监控之ManagementFactory分析

    The ManagementFactory class is a factory class for getting managed beans for the Java platform. This ...