最近写C#串口通信程序,系统是B/S架构。SerialPort类有一个DataReceived事件,用来接收串口返回的数据,但这种方式在C/S架构下很好用,但B/S就不好处理了。所以写了一个同步模式接收返回数据的方法,不使用DataReceived事件。经过测试,可以正常使用。

一、MachineFactory类

为什么使用工厂类:售货机由不止一个厂家提供,接口协议都不一样。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Reflection;
using System.IO;
using System.IO.Ports; namespace IMachineDll
{
/// <summary>
/// 售货机工厂类
/// </summary>
public class MachineFactory
{
/// <summary>
/// 货机接口缓存
/// </summary>
private static Dictionary<string, IMachine> dicMachine = new Dictionary<string, IMachine>();
/// <summary>
/// 锁变量
/// </summary>
public static object _lock = new object(); /// <summary>
/// 创建售货机类
/// </summary>
/// <param name="path">DLL物理路径</param>
/// <param name="dllName">DLL名称(不含扩展名),命名空间必须为DLL名称加“Dll”后缀,类名必须和DLL名称相同</param>
/// <param name="com">串口名称,如:COM1</param>
public static IMachine Create(string path, string dllName, string com)
{
if (!dicMachine.ContainsKey(dllName)
|| dicMachine[dllName] == null)
{
using (FileStream fs = new FileStream(path + dllName + ".dll", FileMode.Open, FileAccess.Read))
{
using (MemoryStream ms = new MemoryStream())
{
byte[] byteArray = new byte[];
while (fs.Read(byteArray, , byteArray.Length) > )
{
ms.Write(byteArray, , byteArray.Length);
} Assembly assembly = Assembly.Load(ms.ToArray());
dicMachine[dllName] = (IMachine)assembly.CreateInstance(dllName + "Dll." + dllName, false, BindingFlags.Default, null, new object[] { com }, null, null);
}
}
} return dicMachine[dllName];
}
}
}

二、Machine类

1、变量与构造函数

/// <summary>
/// 串口资源
/// </summary>
private SerialPort serialPort = null; public Machine(string com)
{
serialPort = new SerialPort(com, , Parity.None, , StopBits.One);
serialPort.ReadBufferSize = ;
serialPort.WriteBufferSize = ;
}

2、向串口发送数据,同步接收返回数据的方法:

/// <summary>
/// 向串口发送数据,读取返回数据
/// </summary>
/// <param name="sendData">发送的数据</param>
/// <returns>返回的数据</returns>
private byte[] ReadPort(byte[] sendData)
{
lock (MachineFactory._lock)
{
//打开连接
if (!serialPort.IsOpen) serialPort.Open(); //发送数据
serialPort.Write(sendData, , sendData.Length); //读取返回数据
DateTime dt = DateTime.Now;
while (serialPort.BytesToRead == )
{
Thread.Sleep(); if (DateTime.Now.Subtract(dt).TotalMilliseconds > ) //如果5秒后仍然无数据返回,则视为超时
{
throw new Exception("主版无响应");
}
}
Thread.Sleep();
byte[] recData = new byte[serialPort.BytesToRead];
serialPort.Read(recData, , recData.Length); //关闭连接
if (serialPort.IsOpen) serialPort.Close(); return recData;
}
}

优化版:

/// <summary>
/// 向串口发送数据,读取返回数据
/// </summary>
/// <param name="sendData">发送的数据</param>
/// <returns>返回的数据</returns>
private byte[] ReadPort(byte[] sendData)
{
lock (MachineFactory._lock)
{
//打开连接
if (!serialPort.IsOpen) serialPort.Open(); //发送数据
serialPort.Write(sendData, , sendData.Length); //读取返回数据
DateTime dt = DateTime.Now;
while (serialPort.BytesToRead < )
{
Thread.Sleep(); if (DateTime.Now.Subtract(dt).TotalMilliseconds > ) //如果5秒后仍然无数据返回,则视为超时
{
throw new Exception("主版无响应");
}
}
List<byte> recList = new List<byte>();
byte[] recData = new byte[serialPort.BytesToRead];
serialPort.Read(recData, , recData.Length);
recList.AddRange(recData);
int length = recData[] + ; //报文数据总长度
while (recList.Count < length)
{
if (serialPort.BytesToRead > )
{
recData = new byte[serialPort.BytesToRead];
serialPort.Read(recData, , recData.Length);
recList.AddRange(recData);
}
Thread.Sleep();
} //关闭连接
if (serialPort.IsOpen) serialPort.Close(); return recList.ToArray();
}
}

3、发送联机指令:

/// <summary>
/// 联机
/// </summary>
/// <param name="msg">传出错误信息</param>
/// <returns>联机是否成功</returns>
public bool Connect(out string msg)
{
byte[] sendData = new byte[] { 0x01, 0x01, 0x00, 0x00 };
CommonUtil.CalCheckCode(sendData);
byte[] recData = ReadPort(sendData); if (recData.Length >=
&& recData[] == 0x01
&& recData[] == 0x02
&& recData[] == 0x00
&& CommonUtil.ValidCheckCode(recData))
{
switch (recData[])
{
case 0x00:
msg = "控制主板正在重启";
return false;
case 0x01:
msg = "联机成功";
return true;
case 0x02:
msg = "控制主板正在维护";
return false;
case 0x03:
msg = "控制主板收到的数据格式不正确";
return false;
default:
msg = "未知状态";
return false;
}
}
else if (IsRunning(recData, out msg) || !IsConnected(recData, out msg))
{
return false;
}
else
{
throw new Exception("货机返回的数据格式不正确");
}
}

三、如何使用

1、Controller层代码(还不完善,仅测试,真实情况是根据硬件信息,确定调用哪个Dll使用哪个串口):

#region 创建售货机接口
/// <summary>
/// 创建售货机接口
/// </summary>
private IMachine CreateMachine()
{
//return MachineFactory.Create(Request.PhysicalApplicationPath + @"\bin\", "Machine", "COM1");
return MachineFactory.Create(@"D:\售药机代码\Reception\Machine\bin\Debug\", "Machine", "COM1");
}
#endregion #region 联机
/// <summary>
/// 联机
/// </summary>
/// <param name="msg">错误信息</param>
/// <returns>联机是否成功</returns>
private bool Connect(out string msg)
{
try
{
IMachine machine = CreateMachine();
DateTime dt1 = DateTime.Now;
while (!machine.Connect(out msg)) //联机
{
if (DateTime.Now.Subtract(dt1).TotalMilliseconds > )
{
msg = "联机超时";
return false;
}
Thread.Sleep();
} return true;
}
catch (Exception ex)
{
msg = ex.Message;
return false;
}
}
#endregion #region 单次联机
/// <summary>
/// 单次联机
/// </summary>
public ActionResult Conn()
{
string msg = null;
Dictionary<string, object> dic = null; try
{
IMachine machine = CreateMachine(); if (machine.Connect(out msg)) //联机成功
{
dic = new Dictionary<string, object>();
dic["ok"] = true;
dic["msg"] = "成功";
return Content(JsonConvert.SerializeObject(dic));
}
else
{
dic = new Dictionary<string, object>();
dic["ok"] = false;
dic["msg"] = "联机失败:" + msg;
return Content(JsonConvert.SerializeObject(dic));
}
}
catch (Exception ex)
{
dic = new Dictionary<string, object>();
dic["ok"] = false;
dic["msg"] = "错误:" + ex.Message;
return Content(JsonConvert.SerializeObject(dic));
}
}
#endregion #region 联机并使能硬纸币器
/// <summary>
/// 联机并使能硬纸币器
/// </summary>
public ActionResult ConnectEnable()
{
string msg = null;
Dictionary<string, object> dic = null; try
{
IMachine machine = CreateMachine(); if (Connect(out msg) && machine.CoinEnable(out msg) && machine.PaperMoneyEnable(out msg)) //联机并使能硬纸币器成功
{
dic = new Dictionary<string, object>();
dic["ok"] = true;
dic["msg"] = "成功";
return Content(JsonConvert.SerializeObject(dic));
}
else
{
dic = new Dictionary<string, object>();
dic["ok"] = false;
dic["msg"] = "硬币器使能失败:" + msg;
return Content(JsonConvert.SerializeObject(dic));
}
}
catch (Exception ex)
{
dic = new Dictionary<string, object>();
dic["ok"] = false;
dic["msg"] = "错误:" + ex.Message;
return Content(JsonConvert.SerializeObject(dic));
}
}
#endregion

2、前台代码:

@{
ViewBag.Title = "货机接口测试";
Layout = null;
} <!DOCTYPE html>
<html>
<head>
<title>@ViewBag.Title</title>
<script type="text/javascript" src="~/Scripts/jquery-1.8.2.min.js"></script>
<script type="text/javascript" src="~/Scripts/LongPolling.js"></script>
</head>
<body>
<div style="padding: 20px;">
<input type="button" value="联机" onclick="connect()" />
<div style="font-size: 20px; line-height: 30px;">
<div style="padding: 20px;">
<span id="msg">&nbsp;</span>
</div>
</div>
</div>
</body>
</html>
<script type="text/javascript">
//联机
function connect() {
commonAjax({
url: "@Url.Content("/MachineInterface/Conn")",
callback: function (data) {
if (data.ok) {
var html = "联机成功";
$("#msg").html(html);
}
else {
alert(data.msg);
}
}
});
}
</script>

C#串口通信—向串口发送数据,同步接收返回数据的更多相关文章

  1. 背水一战 Windows 10 (101) - 应用间通信: 通过协议打开指定的 app 并传递数据以及获取返回数据, 将本 app 沙盒内的文件共享给其他 app 使用

    [源码下载] 背水一战 Windows 10 (101) - 应用间通信: 通过协议打开指定的 app 并传递数据以及获取返回数据, 将本 app 沙盒内的文件共享给其他 app 使用 作者:weba ...

  2. 基于NIO的同步非阻塞编程完整案例,客户端发送请求,服务端获取数据并返回给客户端数据,客户端获取返回数据

    这块还是挺复杂的,挺难理解,但是多练几遍,多看看研究研究其实也就那样,就是一个Selector轮询的过程,这里想要双向通信,客户端和服务端都需要一个Selector,并一直轮询, 直接贴代码: Ser ...

  3. java URL实现调用其他系统发送报文并获取返回数据

    模拟本系统通过Url方式发送报文到目标服务器,并获取返回数据:(实现类) import java.io.BufferedOutputStream; import java.io.BufferedRea ...

  4. WebRequest发送请求并接收返回值

    public string getXmlStr(string hphmcode)         {            string Url = "http://localhost:80 ...

  5. Ajax的GET,POST方法传输数据和接收返回数据

    //首先创建一个Ajax对象 function ajaxFunction(){ var xmlHttp; try{ // Firefox, Opera 8.0+, Safari xmlHttp=new ...

  6. .NET发送请求(get/post/http/https),携带json数据,接收json数据

    C#发送https请求有一点要注意: ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateVa ...

  7. jQuery表单 Ajax向PHP服务端发送文件请求并返回数据

    ImageAjaxUpLoad.htm <!DOCTYPE html> <head> <meta charset='utf-8'> <title>< ...

  8. php用fsockopen实现post提交数据并获得返回数据

    /** * 函数介绍: 用于post方式提交数据 * 输入参数: 完整url, 数据 * 返回值 : 接口返回值 */ function post_it($url, $data = '', $time ...

  9. webservice一片:其中在外线呼叫数据,查看返回数据

    经Android数据被访问,返回的数据(json格公式,object数据类型:strJson) 业务需求:经webservice调用外部暴露数据并返回json数据序列化.阅读到数据库表:[SQ_Eve ...

随机推荐

  1. 系统空闲时间判断&命名验证

    一.系统空闲时间判断 需要一个自动登录注销的功能,当鼠标移动和或者键盘输入的时候认为当前用户在线,否则过了设置时间就自动退出.好在前辈们留下了这样的一个类: MouseKeyBoardOperate: ...

  2. 用avalon实现一个完整的todomvc(带router)

    照着todomvc官网的例子,做了一个avalon版的todos,功能全都有了,而且加了router模块,比司徒大大写的都完善(≧▽≦)/~ js文件整整100行,初次使用avalon,书写过程中绕了 ...

  3. java提高篇(十九)-----数组之二

    前面一节主要介绍了数组的基本概念,对什么是数组稍微深入了一点点,在这篇博文中主要介绍数组的其他方面. 三.性能?请优先考虑数组 在java中有很多方式来存储一系列数据,而且在操作上面比数组方便的多?但 ...

  4. django开发个人简易Blog——数据模型

    提到数据模型,一定要说一下MVC,MVC框架是现代web开发中最流行的开发框架,它将数据与业务逻辑分开,减小了应用之间的高度耦合.个人非常喜欢MVC开发框架,除了具有上述特性,它使得web开发变得非常 ...

  5. Git学习笔记(4)——添加远程仓库,克隆远程库,以及库的推送

    本文记录了远程库的连接和库的克隆和推送. 远程仓库简介 Git是分布式版本控制系统,同一个Git仓库,可以分布到不同的机器上.有一台机器有一个原始版本库,此后,别的机器可以“克隆”这个原始版本库,而且 ...

  6. 说说设计模式~建造者模式(Builder)

    返回目录 建造者模式是我的"设计模式"里创建型模式里的最后一篇,这种模式在实现中,很多架构都用到了,如MVC,MVP,MVVM,它们都是有建造者模式的精髓的,即,创建与表现分享,我 ...

  7. KnockoutJS 3.X API 第四章(13) template绑定

    目的 template绑定(模板绑定)使用渲染模板的结果填充关联的DOM元素. 模板是一种简单方便的方式来构建复杂的UI结构 . 下面介绍两种使用模板绑定的方法: 本地模板是支持foreach,if, ...

  8. 快速入门系列--NOSQL--07MongoDB

    从我第一次听到Nosql这个概念到如今已经走过4个年头了,但仍然没有具体的去做过相应的实践.最近获得一段学习休息时间,购买了Nosql技术实践一书,正在慢慢的学习.在主流观点中,Nosql大体分为4类 ...

  9. Hello Netgen

    Hello Netgen eryar@163.com 摘要Abstract:本文主要介绍如何对下载的Netgen源码进行编译生成Netgen程序和程序开发所需要的库nglib. 关键字Key Word ...

  10. SQL*Loader之CASE7

    CASE7 1. SQL脚本 case7包含两个SQL脚本,一个是删除脚本ulcase7e.sql,一个是创建脚本ulcase7s.sql [oracle@node3 ulcase]$ cat ulc ...