项目中用到的工具,串口modbus协议读写数据。

    public class ModbusHelper
{
private readonly SerialPort _serialPort;
private readonly ILog _logger;
private bool _isReceivedData = false;
private byte _modbusAdd = 0x02;
private int _readTimes = ; public ModbusHelper(string port, int baudRate, byte slave)
{
try
{
_modbusAdd = slave;
_logger = LogManager.GetLogger(this.GetType());
_serialPort = new SerialPort(port, baudRate, Parity.Even, , StopBits.One);
_serialPort.DataReceived += _serialPort_DataReceived;
_logger = LogManager.GetLogger(this.GetType());
}
catch (Exception e)
{
_logger.Error("Modbus串口初始化失败", e);
}
} public bool Open()
{
try
{
if (!_serialPort.IsOpen)
{
_serialPort.Open();
}
_logger.Debug("串口" + _serialPort.PortName + "正常打开");
return true;
}
catch (Exception e)
{
_logger.Error("打开串口打开失败", e);
return false;
}
} public bool Close()
{
try
{
if (_serialPort.IsOpen)
{
_serialPort.Close();
_serialPort.Dispose();
}
_logger.Debug("串口" + _serialPort.PortName + "正常关闭");
return true;
}
catch (Exception err)
{
_logger.Error("关闭串口失败", err);
return false;
}
} private void ResetBuffer()
{
_readTimes = ;
_serialPort.DiscardInBuffer();
} /// <summary>
/// 串口寄存器写值
/// </summary>
/// <param name="beginAddress">开始写入相对地址</param>
/// <param name="data">写入数据</param>
public void WriteSingleWord(short beginAddress, short data)
{
try
{
var writeData = new List<byte>();
writeData.Add(_modbusAdd);
writeData.Add(0x06);
writeData.AddRange(BitConverter.GetBytes(beginAddress).Reverse());
writeData.AddRange(BitConverter.GetBytes(data).Reverse());
byte[] crc = CrcCheck(writeData.ToArray());
writeData.AddRange(crc.Reverse()); if (_serialPort.IsOpen)
{
ResetBuffer();
_serialPort.Write(writeData.ToArray(), , writeData.Count);
_history.Add(writeData);
_serialPort.ReceivedBytesThreshold = ;
while (_readTimes < && !_isReceivedData)
{
_readTimes++;
Thread.Sleep();
}
_logger.Debug("写单字数据成功,数据:" + string.Join(",", writeData));
}
}
catch (Exception err)
{
_logger.Error("写入单字数据时失败", err);
}
} /// <summary>
/// 向寄存器写入多个字
/// </summary>
/// <param name="beginAddress">起始地址(相对地址)</param>
/// <param name="writeLen">写入的字数</param>
/// <param name="data">按字写入数据(字节高地址,字节低地址;字节高地址,字节低地址;)</param>
public void WriteMutiWord(short beginAddress, short writeLen, byte[] data)
{
try
{
var writeData = new List<byte>();
writeData.Add(_modbusAdd);
writeData.Add(0x10);
writeData.AddRange(BitConverter.GetBytes(beginAddress).Reverse());
writeData.AddRange(BitConverter.GetBytes(writeLen).Reverse());
writeData.Add((byte)(writeLen * ));//写入的字节长度
writeData.AddRange(data);
byte[] crc = CrcCheck(writeData.ToArray());
writeData.AddRange(crc.Reverse()); if (_serialPort.IsOpen)
{
ResetBuffer();
_serialPort.Write(writeData.ToArray(), , writeData.Count);
_history.Add(writeData);
_serialPort.ReceivedBytesThreshold = ;
while (_readTimes < && !_isReceivedData)
{
_readTimes++;
Thread.Sleep();
}
_logger.Debug("多字写数据成功,数据:" + string.Join(",", writeData));
}
}
catch (Exception err)
{
_logger.Error("往串口写入数据时失败", err);
}
} /// <summary>
/// 向寄存器写入双字
/// </summary>
/// <param name="beginAddress">起始地址(相对地址)</param>
/// <param name="data">写入数据</param>
public void WriteDoubleWord(short beginAddress, int data)
{
try
{
var writeData = new List<byte>();
writeData.Add(_modbusAdd);
writeData.Add(0x10);
writeData.AddRange(BitConverter.GetBytes(beginAddress).Reverse());
writeData.Add();//寄存器数量高字节
writeData.Add();//寄存器数量低字节
writeData.Add();//写入的字节长度
writeData.AddRange(BitConverter.GetBytes(data).Reverse()); byte[] crc = CrcCheck(writeData.ToArray());
writeData.AddRange(crc.Reverse()); if (_serialPort.IsOpen)
{
ResetBuffer();
_serialPort.Write(writeData.ToArray(), , writeData.Count);
_history.Add(writeData);
_serialPort.ReceivedBytesThreshold = ;
while (_readTimes < && !_isReceivedData)
{
_readTimes++;
Thread.Sleep();
}
_logger.Debug("写双字数据成功,数据:" + string.Join(",", writeData));
}
}
catch (Exception err)
{
_logger.Error("写入双字数据时失败", err);
}
} /// <summary>
/// 串口读取寄存器值
/// </summary>
/// <param name="beginAddress">开始读取相对地址</param>
/// <param name="readLen">读取的字数</param>
/// <returns></returns>
public byte[] Read(short beginAddress, short readLen)
{
try
{
var readData = new List<byte>();
readData.Add(_modbusAdd);
readData.Add(0x03);
readData.AddRange(BitConverter.GetBytes(beginAddress).Reverse());
readData.AddRange(BitConverter.GetBytes(readLen).Reverse());
byte[] crc = CrcCheck(readData.ToArray());
readData.AddRange(crc.Reverse()); if (_serialPort.IsOpen)
{
ResetBuffer();
_serialPort.Write(readData.ToArray(), , readData.Count);
List<string> logdata = new List<string>();
readData.ForEach(x => logdata.Add(Convert.ToString(x, ).ToUpper().PadLeft(,'')));
_logger.Debug("串口写入数据:" + string.Join(" ", logdata));
_serialPort.ReceivedBytesThreshold = + readLen * ;
Thread.Sleep();
while (_serialPort.BytesToRead == && _readTimes < )
{
_readTimes++;
Thread.Sleep();
}
if (_serialPort.BytesToRead > )
{
var bytesToRead = _serialPort.BytesToRead;
byte[] buffer = new byte[bytesToRead];
_serialPort.Read(buffer, , bytesToRead);
_serialPort.DiscardInBuffer(); _logger.Debug("串口返回的数据:" + string.Join(",", buffer));
int readDataLength = bytesToRead - ;//减去开头的modbus地址、功能码、字节数,和结尾的2字节校验位
if (readDataLength > )
{
return buffer.Skip().Take(readDataLength).ToArray();
}
}
return null;
}
return null;
}
catch (Exception err)
{
_logger.Error("写入<读取>指令时失败", err);
return null;
}
} private void _serialPort_DataReceived(object sender, SerialDataReceivedEventArgs e)
{
try
{ }
catch (Exception err)
{
_logger.Error("读取串口数据失败", err);
}
} /// <summary>
/// 获取Modbus校验值
/// </summary>
/// <param name="data">数据字节流</param>
/// <returns>返回2byte,byte0高位,byte1低位</returns>
private byte[] CrcCheck(byte[] data)
{
try
{
byte CRC16Lo;
byte CRC16Hi;
byte CL; byte CH;
byte SaveHi; byte SaveLo;
byte[] tmpData;
int Flag;
CRC16Lo = 0xFF;
CRC16Hi = 0xFF;
CL = 0x01;
CH = 0xA0;
tmpData = data;
for (int i = ; i < tmpData.Length; i++)
{
CRC16Lo = (byte)(CRC16Lo ^ tmpData[i]);
for (Flag = ; Flag <= ; Flag++)
{
SaveHi = CRC16Hi;
SaveLo = CRC16Lo;
CRC16Hi = (byte)(CRC16Hi >> );
CRC16Lo = (byte)(CRC16Lo >> );
if ((SaveHi & 0x01) == 0x01)
{
CRC16Lo = (byte)(CRC16Lo | 0x80);
} if ((SaveLo & 0x01) == 0x01)
{
CRC16Hi = (byte)(CRC16Hi ^ CH);
CRC16Lo = (byte)(CRC16Lo ^ CL);
}
}
}
byte[] ReturnData = new byte[];
ReturnData[] = CRC16Hi;
ReturnData[] = CRC16Lo;
return ReturnData;
}
catch (Exception err)
{
_logger.Error("计算crc校验值出错", err);
return null;
}
} private readonly List<List<byte>> _history = new List<List<byte>>(); public void ClearHistory()
{
_history.Clear();
} public void WriteHistory()
{
foreach (List<byte> item in _history)
{
_serialPort.Write(item.ToArray(), , item.Count);
_serialPort.ReceivedBytesThreshold = ;
while (_readTimes < && !_isReceivedData)
{
_readTimes++;
Thread.Sleep();
}
}
} }

Modbus Com SerialPort的更多相关文章

  1. Modbus RTU 通信工具设计(转)

    Modbus RTU 通信工具设计 Modbus 是一个工业上常用的通讯协议.一种通讯约定. ModBus 协议是应用层报文传输协议(OSI 模型第7层),它定义了一个与通信层无关的协议数据单元(PD ...

  2. C# MODBUS协议 上位机(转)

    源:C# MODBUS协议 上位机 C#写了一款上位机监控软件,基于MODBUS_RTU协议. 软件的基本结构: 采用定时器(Timer控件)为时间片. 串口采用serialPort1_DataRec ...

  3. Modbus软件开发实战指南 之 开发自己的Modbus Poll工具 - 1

    在开发Modbus程序的过程中,也可以发现经常需要使用诸如Modbus Poll和Modbus Slave等辅助调试工具, 用于验证MODBUS通讯消息是否正确.但是,Modbus Poll和Modb ...

  4. 基于Modbus的C#串口调试开发

    说明:本文主要研究的是使用C# WinForm开发的串口调试软件(其中包含Modbus协议相关操作).Modbus相关协议可以查阅百度文库等,可参考: <http://wenku.baidu.c ...

  5. modbus串口通讯C#

    简介 公司给的一个小任务,这篇文章进行详细讲解 题目: modbus串口通讯 主要内容如下: 1.实现使用modbus通讯规约的测试软件: 2.具有通信超时功能: 3.分主站从站,并能编辑报文.生成报 ...

  6. Modbus tcp 格式说明 通讯机制 附C#测试工具用于学习,测试

    前言: 之前的博客介绍了如何用C#来读写modbus tcp服务器的数据,文章:http://www.cnblogs.com/dathlin/p/7885368.html 当然也有如何创建一个服务器文 ...

  7. MODBUS协议相关代码(CRC验证 客户端程序)

    Modbus协议是一种已广泛应用于当今工业控制领域的通用通讯协议.通过此协议,控制器相互之间.或控制器经由网络(如以太网)可以和其它设备之间进行通信.Modbus协议使用的是主从通讯技术,即由主设备主 ...

  8. C# MODBUS 通信

    背景 电厂有多组监控设备,需要在指定的设备上显示某些数据(其他设备对接过来的).通信协议是modbus主从结构. 源码: http://download.csdn.net/download/wolf1 ...

  9. C#Modbus Rtu的实现

    Modbus Rtu的实现与Modbus Tcp的实现类似 C#ModBus Tcp的学习及Master的实现 我们还是需要借用一个开源库NModbus4,在vs中.打开NuGet管理器.安装NMod ...

随机推荐

  1. 重装SQL前,一定要把SQL2005、SQL2008之类的彻底删除干净

    0.预备 如果你曾删除过VS2010或者VS2008之类的,同理也要照此方法删除 1.步骤,顺序无妨 卸载程序:控制面板---查找SQL..NET   删除干净 停掉SQL的所有服务:  计算机--管 ...

  2. Python脱产8期 Day15 2019/4/30

    一 生成器send方法 1.send的工作原理# 1.send发生信息给当前停止的yield# 2.再去调用__next__()方法,生成器接着往下指向,返回下一个yield值并停止 2.例: per ...

  3. “全栈2019”Java第九十三章:内部类应用场景(迭代器设计模式)

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  4. kali linux之Audacity

    常用音频隐写工具 安装: sudo apt install audacity 初次打开的界面 看波的宽度分辨长短音 比较细的就是短音,代表".",比较粗的就是长音,代表" ...

  5. [arc079f] Namori Grundy 分类讨论

    Description 给给全有一个NN个点NN条边的有向图,点的的编号从11到NN 给给全的图有NN条边,形如:(p1,1),(p2,2),...,(pN,N)(p1,1),(p2,2),...,( ...

  6. javascript简要笔记

      零. 数据   0. 变量 分为字符串,数字,undefined, null,对象 undefined类型是只声明了变量,但是没赋值 可以使用typeof()函数来查看变量类型   例子1 var ...

  7. SSM搭建

    SSM搭建 SSM(Spring+SpringMVC+MyBatis)框架集由Spring.SpringMVC.MyBatis三个开源框架整合而成,常作为数据源较简单的web项目的框架.. Sprin ...

  8. Microsoft Visual C++ 14.0 is required

    building 'twisted.test.raiser' extensionerror: Microsoft Visual C++ 14.0 is required. Get it with &q ...

  9. leetcode-201-数字范围按位与

    题目描述: 给定范围 [m, n],其中 0 <= m <= n <= 2147483647,返回此范围内所有数字的按位与(包含 m, n 两端点). 示例 1: 输入: [5,7] ...

  10. 架构师养成记--29.redis开篇

    主要有从下几点讲解 NOSQL(Redis) 简介.redis安装与部署 Redis基础事件类型详解 Redis高级命令 Redis与java的使用 Redis集群搭建 Redis集群与spring的 ...