.Net Core跨平台应用研究-CustomSerialPort

-增强型跨平台串口类库

摘要

在使用SerialPort进行串口协议解析过程中,经常遇到接收单帧协议数据串口接收事件多次触发,协议解析麻烦的问题。针对此情况,基于开源跨平台串口类库SerialPortStrem进行了进一步封装,实现了一种接收超时响应事件机制,简化串口通讯的使用。

引言

最近,写了一篇博文《.net core跨平台应用研究-串口篇》得到了一些园友的好评,文中介绍了在跨平台应用研究过程中,在dotnet core下使用SerialPort类库在linux下不能支持的踩坑经历及解决办法。

因网上关于SerialPort类库使用的相关文章较多,在该文中,对串口类库的使用,一笔带过。但在实际使用,使用过SerialPort类库的同学,可能遇到过在数据接收时,由于数据接收事件的触发具有不确定性,很多时候,一帧通讯协议数据,会多次触发,造成程序处理协议数据较为麻烦的问题。

为简化串口通讯类库的使用,笔者结合自己的相关经验,封装了一个自定义增强型跨平台串口类库,以解决一帧协议数据,多次触发的问题。

基础类库的选择

由于考虑的是跨平台应用,SerialPort类库并不支持linux系统(在前一篇文章中已介绍过踩坑经历),笔者选用了SerialPortStream类库进行封装。

该类库支持windows系统和Linux系统,但在Linux系统下运行,需要额外编译目标平台支持库并进行相关环境配置。

相关编译配置说明在https://github.com/jcurl/SerialPortStream已有介绍,也可参考本人的拙作《.net core跨平台应用研究-串口篇》

类库的实现

创建跨平台类库

为了支持跨平台,我们使用Visual Studio 2017创建一个基于.NET Standard的类库。

NET Standard是一项API规范,每一个特定的版本,都定义了必须实现的基类库。

.NET Core是一个托管框架,针对构建控制台、云、ASP.NET Core和UWP应用程序进行了优化。

每一种托管实现(如.NET Core、.NET Framework或Xamarin)都必须遵循.NET Standard实现基类库(BCL)。

关于NET Standard和跨平台的详细说明在此:

https://zhuanlan.zhihu.com/p/30081607

笔者也不再啰嗦呵。

实现机制/条件

通常串口通讯中,发送数据后,会有一段时间用于等待接收方应答,如此一来,两次数据发送之间,必然会有一定的时间间隔。如ModbusRTU协议就规定,两次数据报文发送之间,需要等待超过发送4个字节以上的间隔时间。

笔者在单片机以及实时性较高的嵌入式系统中,为处理串口接收与协议的无关性,通常采用数据帧接收超时来处理数据帧的接收。根据串口通讯的速率计算出两次通讯之间所需要超时间隔,取两倍超时间隔时间作为超时参数,每接收到一个字节,将数据放入缓冲区并进行计时,当最后一个字节的接收时间超过超时时间,返回接收数据并清空缓存,一次完整接收完成(DMA接收方式不在此讨论)。

.net core跨平台实现

在自定义的串口类中,订阅基础串口类数据接收事件,在接收事件每次触发后,读出当前可用的缓冲数据到自定义缓冲区,同时,标记最后接收时间Tick为当前系统Tick。判断是否开启了接收超时处理线程,如未开启,则开启一个接收超时处理线程。

接收超时处理线程中,以一个较小的时间间隔进行判断,如果最后接收时间与当前时间之间的间隔小于设置值(默认128ms),休眠一段时间(默认16ms)后循环检查。如间隔时间大于设定值,触发外部接收订阅事件,传出接收到的数据,退出超时处理线程。

此处应有流程图。呵呵,懒得画了,大家自行脑补吧。 ^_^

在windows系统或linux系统中,因系统的多任务处理的特性,系统实时性较差,通常50ms以下时间间隔的定时任务,较大程度会出现不可靠的情况(任务执行时间都有可能超过调用间隔时间)。

因此,默认超时时间间隔设置为128ms。也可根据实际使用情况调整,但最小间隔不宜低于64ms。

注:此处为个人经验和理解,如不认同,请直接忽视。

主要代码

串口接收事件代码:

         protected void Sp_DataReceived(object sender, SerialDataReceivedEventArgs e)
{
int canReadBytesLen = ;
if (ReceiveTimeoutEnable)
{
while (sp.BytesToRead > )
{
canReadBytesLen = sp.BytesToRead;
if (receiveDatalen + canReadBytesLen > BufSize)
{
receiveDatalen = ;
throw new Exception("Serial port receives buffer overflow!");
}
var receiveLen = sp.Read(recviceBuffer, receiveDatalen, canReadBytesLen);
if (receiveLen != canReadBytesLen)
{
receiveDatalen = ;
throw new Exception("Serial port receives exception!");
}
//Array.Copy(recviceBuffer, 0, receivedBytes, receiveDatalen, receiveLen);
receiveDatalen += receiveLen;
lastReceiveTick = Environment.TickCount;
if (!TimeoutCheckThreadIsWork)
{
TimeoutCheckThreadIsWork = true;
Thread thread = new Thread(ReceiveTimeoutCheckFunc)
{
Name = "ComReceiveTimeoutCheckThread"
};
thread.Start();
}
}
}
else
{
if (ReceivedEvent != null)
{
// 获取字节长度
int bytesNum = sp.BytesToRead;
if (bytesNum == )
return;
// 创建字节数组
byte[] resultBuffer = new byte[bytesNum]; int i = ;
while (i < bytesNum)
{
// 读取数据到缓冲区
int j = sp.Read(recviceBuffer, i, bytesNum - i);
i += j;
}
Array.Copy(recviceBuffer, , resultBuffer, , i);
ReceivedEvent(this, resultBuffer);
//System.Diagnostics.Debug.WriteLine("len " + i.ToString() + " " + ByteToHexStr(resultBuffer));
}
//Array.Clear (receivedBytes,0,receivedBytes.Length );
receiveDatalen = ;
}
}

接收超时处理线程代码:

         /// <summary>
/// 超时返回数据处理线程方法
/// </summary>
protected void ReceiveTimeoutCheckFunc()
{
while (TimeoutCheckThreadIsWork)
{
if (Environment.TickCount - lastReceiveTick > ReceiveTimeout)
{
if (ReceivedEvent != null)
{
byte[] returnBytes = new byte[receiveDatalen];
Array.Copy(recviceBuffer, , returnBytes, , receiveDatalen);
ReceivedEvent(this, returnBytes);
}
//Array.Clear (receivedBytes,0,receivedBytes.Length );
receiveDatalen = ;
TimeoutCheckThreadIsWork = false;
}
else
Thread.Sleep();
}
}

创建.net core控制台程序

为验证我们的类库是否能够正常工作,我们创建一个使用类库的.net core控制台程序。

为啥选择dotnet core,原因很简单,跨平台。本程序分别需在windows和linux系统下进行运行测试。

控制台程序主要实现以下功能:

  • 显示系统信息(系统标识、程序标识等)
  • 列举系统可用串口资源
  • 选择串口
  • 打开串口/关闭串口
  • 串口测试(打开/发送/关闭)
         static void Main(string[] args)
{
SetLibPath();
ShowWelcome(); GetPortNames();
ShowPortNames(); if (serailports.Length == )
{
Console.WriteLine($"Press any key to exit");
Console.ReadKey(); return;
}
#if RunIsService
RunService();
#endif bool quit = false;
while (!quit)
{
Console.WriteLine("\r\nPlease Input command Key\r\n");
Console.WriteLine("p:Show SerialPort List");
Console.WriteLine($"t:Test Uart:\"{selectedComPort}\"");
Console.WriteLine($"o:Open Uart:\"{selectedComPort}\"");
Console.WriteLine($"c:Close Uart:\"{selectedComPort}\"");
Console.WriteLine("n:select next serial port");
Console.WriteLine("q:exit app");
Console.WriteLine();
var key = Console.ReadKey().KeyChar;
Console.WriteLine(); switch (key)
{
case (Char):
case 'q':
case 'Q':
quit = true;
break;
case 's':
ShowWelcome();
break;
case 'p':
ShowPortNames();
break;
case 'n':
SelectSerialPort();
break;
case 't':
TestUart(selectedComPort);
break;
case 'w':
TestWinUart(selectedComPort);
break;
case 'o':
OpenUart(selectedComPort);
break;
case 'c':
CloseUart();
break;
}
}
}

笔者使用类库是直接引用类库项目,大家需要使用的话,可在解决方案资源管理器中,项目的依赖项上点击右键

在NuGet包管理器中,搜索SerialPort或flyfire即可找到并安装本类库。

类库地址

类库地址:https://www.nuget.org/packages/flyfire.CustomSerialPort

跨平台测试

Windows测试输出界面

ubuntu测试输出界面

源码地址

类库源码与例程地址:https://github.com/flyfire-cn/flyfire.CustomSerialPort

有需要的同学,请自行获取。

.Net Core跨平台应用研究-CustomSerialPort(增强型跨平台串口类库)的更多相关文章

  1. .Net Core跨平台应用研究-HelloArm(串口篇)

    引言 为了验证采用dotnet core技术开发的物联网设备数据采集接入服务应用是否能在高性价比的linux嵌入式平台运行,针对dotnet core应用程序进行嵌入式linux环境的发布部署运行验证 ...

  2. .net core跨平台应用研究-ubuntu core下配置.net core运行时

    引言 年初研究了一阵子.net core跨平台应用,先后发表了几篇应用研究的文章.因工作原因,忙于项目上线,有一阵子没来博客园写文章了.最近项目基本收尾,抽空翻了下自己的博客,廖廖几篇文章,真让人汗颜 ...

  3. .Net Core跨平台应用研究-HelloDDNS(动态域名篇)

    .Net Core跨平台应用研究-HelloDDNS -玩转DDNS 摘要 为解决自己搭建的内网服务器需要域名而因没有超级用户密码不能开启光猫内置DDNS功能的问题,自己动手,基于.net core, ...

  4. ASP.NET Core环境并运行 继续跨平台

    ASP.NET Core环境并运行 继续跨平台 无需安装mono,在Linux(Ubuntu)下搭建ASP.NET Core环境 继续.NET跨平台 上一篇:使用VS Code开发ASP.NET Co ...

  5. NET Core 2.0 微服务跨平台实践

    NET Core 2.0 微服务跨平台实践 相关博文: Ubuntu 简单安装 Docker Mac OS.Ubuntu 安装及使用 Consul Consul 服务注册与服务发现 Fabio 安装和 ...

  6. .NET Core 系列5 :使用 Nuget打包类库

    NuGet是个开源项目,项目包括 NuGet VS插件/NuGet Explorer/NuGetServer/NuGet命令行等项目,.NET Core项目完全使用Nuget 管理组件之间的依赖关系, ...

  7. .NET Core 首例 Office 开源跨平台组件(NPOI Core)

    前言 最近项目中,需要使用到 Excel 导出,找了一圈发现没有适用于 .NET Core的,不依赖Office和操作系统限制的 Office 组件,于是萌生了把 NPOI 适配并移植到 .NET C ...

  8. 拥抱.NET Core,如何开发一个跨平台类库 (1)

    在此前的文章中详细介绍了使用.NET Core的基本知识,如果还没有看,可以先去了解“拥抱.NET Core,学习.NET Core的基础知识补遗”,以便接下来的阅读. 在本文将介绍如何配置类库项目支 ...

  9. 聊聊ASP.NET Core默认提供的这个跨平台的服务器——KestrelServer

    跨平台是ASP.NET Core一个显著的特性,而KestrelServer是目前微软推出了唯一一个能够真正跨平台的Server.KestrelServer利用一个名为KestrelEngine的网络 ...

随机推荐

  1. ViewBag赋值Html格式值

    今天再给自己总结一下,关于ViewBag赋值Html格式值,但是在web页显示不正常; 例如,ViewBag.Content = "<p>你好,我现在测试一个东西.</p& ...

  2. Linux文件属性描述

    mtime -- modify time 修改时间 硬链接硬链接(hard link, 也称链接)就是一个文件的一个或多个文件名.再说白点,所谓链接无非是把文件名和计算机文件系统使用的节点号链接起来. ...

  3. 关于查询中查询无果,也不报错,inpout标签中的value属性为‘ ’的判断问题

    首先当我们标签中vlue属性可能为' '时,我们一定要在后端进行判断过滤,不然查询会什么都查不出来的,遇到的问题如下 例子如下: 这是一个easyui 中的下拉选,效果如下 当我们默认查询全部时,后台 ...

  4. SSRF漏洞浅析

    大部分web应用都提供了从其他的服务器上获取数据的功能,如使用用户指定的URL,web应用可以获取图片,下载文件,读取文件内容等.如果服务端提供了从其他服务器应用获取数据的功能且没有对目标地址做过滤与 ...

  5. 修改XAMPP的默认根目录

    XAMPP安装完成后,默认根目录路径是C:\xampp\htdocs,如果想要在服务器下运行文件就必须把该文件copy到C:\xampp\htdocs下.超麻烦不说,公司代码总不能放进去运行吧...所 ...

  6. 如何使用idea给系统平台添加子应用和应用的模块

    1.添加模块smartcity-portal,由于模块是在smartcity-framework工程下的,所以按照图片所示添加 2.smartcity-portal模块添加完成后,由于portal-d ...

  7. Unity UGUI Layout自动排版组件用法介绍

    Unity UGUI布局组件 本文提供全流程,中文翻译. Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 -- 高分辨率用户请根据需求调整网页缩放比例) Chinar -- 心分享 ...

  8. 自定义EL函数(以将字母转为大写为例)

    Step1 定义一个类:StringFunction.java 主要作用是来提供转大写的方法; public class StringFunction { public static String t ...

  9. python基础(九)

    一.私有 class DB: port = 3306 #类变量 def __init__(self): self.host = '127.0.0.1' self.__user = 'root' #实例 ...

  10. 51Nod - 1433 0和5 找规律

    小K手中有n张牌,每张牌上有一个一位数的数,这个字数不是0就是5.小K从这些牌在抽出任意张(不能抽0张),排成一行这样就组成了一个数.使得这个数尽可能大,而且可以被90整除. 注意: 1.这个数没有前 ...