C#实现ModbusRTU详解【四】—— 通讯Demo
完成ModbusRTU Winfrom Demo
前言
通过前面的三篇文章,我们已经基本了解ModbusRTU是个什么东西,以及如何通过C#生成需要的八种常用的通讯报文了,接下来我们就需要完整地实现ModbusRTU通讯了。
传送门:
接下来我们将会使用前面生成的读写报文,实现完整的ModbusRTU通讯。
程序思路
第一步:我们需要确定使用什么样的程序来完成我们的通讯Demo,本文示例将使用Winform,也可根据个人喜好选择WPF或者其它项目类型。确定好之后创建对应的项目,并完成窗体的UI设计(功能优先)。窗体上应有通讯参数设定、通讯开关、报文参数设定、读写模式选择、报文收发显示、报文发送按钮以及接收到的报文的解析等内容。
第二步:通过前面的文章,我们已经知道了,ModbusRTU通讯一般是以串口为通讯介质的,所以我们需要一个串口通讯类,它应该包括串口的参数设定、打开与关闭、收发数据的方法或事件。
第三步:引入我们已经写好的报文生成方法,以及解析报文的方法,并根据实际需要进行结构调整。
第四步:创建所需的窗体事件,并在窗体事件中调用已经添加完成的通讯方法和报文生成方法。
第五步:测试程序是否能正常运行,并根据出现的问题修改程序,并尽量优化使用体验。
明确了思路之后,我们就可以开始搭建我们的ModbusRTU Demo了。
创建项目
新建一个Winform程序,根据自己喜好创建布局,参考如下:
然后在项目中创建两个文件夹,分别用以存放通讯类及报文类,如下图所示:
引入通讯类
根据C# 串口通讯这篇文章,我们可以得知C#如何搭建串口通讯,此处不再赘述搭建方法,以下为SerialPortHelper的代码:
using System;
using System.Collections.Generic;
using System.IO.Ports;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace ModbusRTUDemo.Communication
{
/// <summary>
/// 自定义串口消息接收事件委托
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
public delegate void ReceiveDataEventHandler(object sender, ReceiveDataEventArg e);
class SerialPortHelper
{
/// <summary>
/// 自定义串口消息接收事件
/// </summary>
public event ReceiveDataEventHandler ReceiveDataEvent;
//串口字段
private SerialPort serialPort;
/// <summary>
/// 构造函数
/// </summary>
public SerialPortHelper()
{
serialPort = new SerialPort();
}
/// <summary>
/// 串口状态
/// </summary>
public bool Status { get => serialPort.IsOpen; }
/// <summary>
/// 获取当前计算机所有的串行端口名
/// </summary>
/// <returns></returns>
public static string[] GetPortArray()
{
return SerialPort.GetPortNames();
}
/// <summary>
/// 串口参数
/// </summary>
public void SetSerialPort(string portName, int baudrate, Parity parity, int databits, StopBits stopBits)
{
//端口名
serialPort.PortName = portName;
//波特率
serialPort.BaudRate = baudrate;
//奇偶校验
serialPort.Parity = parity;
//数据位
serialPort.DataBits = databits;
//停止位
serialPort.StopBits = stopBits;
//串口接收数据事件
serialPort.DataReceived += ReceiveDataMethod;
}
/// <summary>
/// 打开串口
/// </summary>
public void Open()
{
//打开串口
serialPort.Open();
}
/// <summary>
/// 关闭串口
/// </summary>
public void Close()
{
serialPort.Close();
}
/// <summary>
/// 发送数据
/// </summary>
/// <param name="data">要发送的数据</param>
public void SendDataMethod(byte[] data)
{
//获取串口状态,true为已打开,false为未打开
bool isOpen = serialPort.IsOpen;
if (!isOpen)
{
Open();
}
//发送字节数组
//参数1:包含要写入端口的数据的字节数组。
//参数2:参数中从零开始的字节偏移量,从此处开始将字节复制到端口。
//参数3:要写入的字节数。
serialPort.Write(data, 0, data.Length);
}
/// <summary>
/// 发送数据
/// </summary>
/// <param name="data">要发送的数据</param>
public void SendDataMethod(string data)
{
//获取串口状态,true为已打开,false为未打开
bool isOpen = serialPort.IsOpen;
if (!isOpen)
{
Open();
}
//直接发送字符串
serialPort.Write(data);
}
/// <summary>
/// 串口接收到数据触发此方法进行数据读取
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void ReceiveDataMethod(object sender, SerialDataReceivedEventArgs e)
{
ReceiveDataEventArg arg = new ReceiveDataEventArg();
//读取串口缓冲区的字节数据
arg.Data = new byte[serialPort.BytesToRead];
serialPort.Read(arg.Data, 0, serialPort.BytesToRead);
//触发自定义消息接收事件,把串口数据发送出去
if (ReceiveDataEvent != null && arg.Data.Length != 0)
{
ReceiveDataEvent.Invoke(null, arg);
}
}
}
}
ReceiveDataEventArg类:
/// <summary>
/// 串口接收数据事件的参数
/// </summary>
public class ReceiveDataEventArg : EventArgs
{
/// <summary>
/// 串口接收到的数据
/// </summary>
public byte[] Data { get; set; }
}
通讯的实现都是常规操作,但是有一个需要注意的地方,为了满足这个类的通用性,我们为通讯方法创建了一个自定义的消息接收事件。因为我们的SerialPort类的DataReceived事件的参数是不带接收到的数据的,所以当触发DataReceived事件时,需要手动去从缓冲区去获取接收的数据。因此我们自定义了一个消息接收的事件,当串口接收到数据触发DataReceived事件时,我们在DataReceived触发的方法里去获取串口接收到的数据,然后再通过自定义事件把这个数据用参数传递出去,这样的好处就是使用这个通讯类的时候,我们不需要关注如何去获取数据。
引入报文类
首先还是我们的读写模式的枚举类型和校验码类:
ReadMode:
public enum ReadMode
{
Read01 = 0x01,
Read02 = 0x02,
Read03 = 0x03,
Read04 = 0x04,
}
WriteMode:
public enum WriteMode
{
Write01 = 0x05,
Write03 = 0x06,
Write01s = 0x0F,
Write03s = 0x10
}
CheckSum:
class CheckSum
{
/// <summary>
/// CRC16校验码计算
/// </summary>
/// <param name="data">要计算的报文</param>
/// <returns></returns>
public static byte[] CRC16(byte[] data)
{
int len = data.Length;
if (len > 0)
{
ushort crc = 0xFFFF;
for (int i = 0; i < len; i++)
{
crc = (ushort)(crc ^ (data[i]));
for (int j = 0; j < 8; j++)
{
crc = (crc & 1) != 0 ? (ushort)((crc >> 1) ^ 0xA001) : (ushort)(crc >> 1);
}
}
byte hi = (byte)((crc & 0xFF00) >> 8); //高位置
byte lo = (byte)(crc & 0x00FF); //低位置
return BitConverter.IsLittleEndian ? new byte[] { lo, hi } : new byte[] { hi, lo };
}
return new byte[] { 0, 0 };
}
/// <summary>
/// CRC16校验码计算
/// </summary>
/// <param name="data">要计算的报文</param>
/// <returns></returns>
public static byte[] CRC16(List<byte> data)
{
int len = data.Count;
if (len > 0)
{
ushort crc = 0xFFFF;
for (int i = 0; i < len; i++)
{
crc = (ushort)(crc ^ (data[i]));
for (int j = 0; j < 8; j++)
{
crc = (crc & 1) != 0 ? (ushort)((crc >> 1) ^ 0xA001) : (ushort)(crc >> 1);
}
}
byte hi = (byte)((crc & 0xFF00) >> 8); //高位置
byte lo = (byte)(crc & 0x00FF); //低位置
return BitConverter.IsLittleEndian ? new byte[] { lo, hi } : new byte[] { hi, lo };
}
return new byte[] { 0, 0 };
}
}
接着就是报文生成的类(MessageGenerationModule):
class MessageGenerationModule
{
/// <summary>
/// 生成读取报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="mode">读取模式</param>
/// <param name="startAdr">起始地址</param>
/// <param name="length">读取长度</param>
/// <returns></returns>
public static byte[] MessageGeneration(int slaveStation, ReadMode mode, short startAdr, short length)
{
return GetReadMessage(slaveStation, mode, startAdr, length);
}
/// <summary>
/// 生成写入报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="mode">写入模式</param>
/// <param name="startAdr">起始地址</param>
/// <param name="value">写入值</param>
/// <returns></returns>
//public static byte[] MessageGeneration(int slaveStation, WriteMode mode, short startAdr, object value)
//{
// //C# 8.0以下版本的写法:
// switch (mode)
// {
// case WriteMode.Write01:
// return GetSingleBoolWriteMessage(slaveStation, startAdr, (bool)value);
// case WriteMode.Write03:
// return GetSingleDataWriteMessage(slaveStation, startAdr, Convert.ToInt16(value));
// case WriteMode.Write01s:
// return GetArrayBoolWriteMessage(slaveStation, startAdr, (IEnumerable<bool>)value);
// case WriteMode.Write03s:
// return GetArrayDataWriteMessage(slaveStation, startAdr, (IEnumerable<short>)value);
// default:
// return null;
// }
//}
public static byte[] MessageGeneration(int slaveStation, WriteMode mode, short startAdr, object value) => mode switch
{
//C# 8.0开始支持此写法,具体可查阅微软官方文档,switch表达式
WriteMode.Write01 => GetSingleBoolWriteMessage(slaveStation, startAdr, (bool)value),
WriteMode.Write03 => GetSingleDataWriteMessage(slaveStation, startAdr, Convert.ToInt16(value)),
WriteMode.Write01s => GetArrayBoolWriteMessage(slaveStation, startAdr, (IEnumerable<bool>)value),
WriteMode.Write03s => GetArrayDataWriteMessage(slaveStation, startAdr, (IEnumerable<short>)value),
_ => throw new ArgumentOutOfRangeException(nameof(mode), $"Not expected WriteMode value: {mode}"),
};
/// <summary>
/// 获取读取数据请求报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="mode">读取模式</param>
/// <param name="startAdr">起始地址</param>
/// <param name="length">读取长度</param>
/// <returns></returns>
private static byte[] GetReadMessage(int slaveStation, ReadMode mode, short startAdr, short length)
{
//定义临时字节列表
List<byte> temp = new List<byte>();
//依次放入头两位字节(站地址和读取模式)
temp.Add((byte)slaveStation);
temp.Add((byte)mode);
//获取起始地址及读取长度
byte[] start = BitConverter.GetBytes(startAdr);
byte[] count = BitConverter.GetBytes(length);
//判断系统是否为小端存储
//如果为true,BitConverter.GetBytes方法会返回低字节在前,高字节在后的字节数组,
//而ModbusRTU则需要高字节在前,低字节在后,所以需要做一次反转操作。
if (BitConverter.IsLittleEndian)
{
Array.Reverse(start);
Array.Reverse(count);
}
//依次放入起始地址和读取长度
temp.AddRange(start);
temp.AddRange(count);
//获取校验码并在最后放入
temp.AddRange(CheckSum.CRC16(temp));
return temp.ToArray();
}
/// <summary>
/// 获取写入单个线圈的报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="startAdr">线圈地址</param>
/// <param name="value">写入值</param>
/// <returns>写入单个线圈的报文</returns>
private static byte[] GetSingleBoolWriteMessage(int slaveStation, short startAdr, bool value)
{
//创建字节列表
List<byte> temp = new List<byte>();
//插入站地址及功能码
temp.Add((byte)slaveStation);
temp.Add(0x05);
//获取线圈地址
byte[] start = BitConverter.GetBytes(startAdr);
//根据计算机大小端存储方式进行高低字节转换
if (BitConverter.IsLittleEndian) Array.Reverse(start);
//插入线圈地址
temp.Add(start[0]);
temp.Add(start[1]);
//插入写入值
temp.Add((byte)(value ? 0xFF : 0x00));
temp.Add(0x00);
//转换为字节数组
byte[] result = temp.ToArray();
//计算校验码并拼接,返回最后的报文结果
return result.Concat(CheckSum.CRC16(temp)).ToArray();
}
/// <summary>
/// 获取写入单个寄存器的报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="startAdr">寄存器地址</param>
/// <param name="value">写入值</param>
/// <returns>写入单个寄存器的报文</returns>
private static byte[] GetSingleDataWriteMessage(int slaveStation, short startAdr, short value)
{
//从站地址
byte station = (byte)slaveStation;
//功能码
byte type = 0x06;
//寄存器地址
byte[] start = BitConverter.GetBytes(startAdr);
//值
byte[] valueBytes = BitConverter.GetBytes(value);
//根据计算机大小端存储方式进行高低字节转换
if (BitConverter.IsLittleEndian)
{
Array.Reverse(start);
Array.Reverse(valueBytes);
}
//拼接报文
byte[] result = new byte[] { station, type };
result = result.Concat(start.Concat(valueBytes).ToArray()).ToArray();
//计算校验码并拼接,返回最后的报文结果
return result.Concat(CheckSum.CRC16(result)).ToArray();
}
/// <summary>
/// 获取写入多个线圈的报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="startAdr">起始地址</param>
/// <param name="value">写入值</param>
/// <returns>写入多个线圈的报文</returns>
private static byte[] GetArrayBoolWriteMessage(int slaveStation, short startAdr, IEnumerable<bool> value)
{
//定义报文临时存储字节集合
List<byte> tempList = new List<byte>();
//插入从站地址
tempList.Add((byte)slaveStation);
//插入功能码
tempList.Add(0x0F);
//获取起始地址
byte[] start = BitConverter.GetBytes(startAdr);
//获取写入线圈数量
byte[] length = BitConverter.GetBytes(Convert.ToInt16(value.Count()));
//根据计算机大小端存储方式进行高低字节转换
if (BitConverter.IsLittleEndian)
{
Array.Reverse(start);
Array.Reverse(length);
}
//插入起始地址
tempList.Add(start[0]);
tempList.Add(start[1]);
//插入写入线圈数量
tempList.Add(length[0]);
tempList.Add(length[1]);
//定义写入值字节集合
List<byte> valueTemp = new List<byte>();
//由于一个字节只有八个位,所以如果需要写入的值超过了八个,
//则需要生成一个新的字节用以存储,
//所以循环截取输入的值,然后生成对应的写入值字节
for (int i = 0; i < value.Count(); i += 8)
{
//写入值字节临时字节集合
List<bool> temp = value.Skip(i).Take(8).ToList();
//剩余位不足八个,则把剩下的所有位都放到同一个字节里
if (temp.Count != 8)
{
//取余获取剩余的位的数量
int m = value.Count() % 8;
//截取位放入临时字节集合中
temp = value.Skip(i).Take(m).ToList();
}
//获取位生成的写入值字节
byte tempByte = GetBitArray(temp);
//将生成的写入值字节拼接到写入值字节集合中
valueTemp.Add(tempByte);
}
//获取写入值的字节数
byte bytecount = (byte)valueTemp.Count;
//插入写入值的字节数
tempList.Add(bytecount);
//插入值字节集合
tempList.AddRange(valueTemp);
//根据报文字节集合计算CRC16校验码,并拼接到最后,然后转换为字节数组并返回
return tempList.Concat(CheckSum.CRC16(tempList)).ToArray();
}
/// <summary>
/// 获取写入多个寄存器的报文
/// </summary>
/// <param name="slaveStation">从站地址</param>
/// <param name="startAdr">起始地址</param>
/// <param name="value">写入值</param>
/// <returns>写入多个寄存器的报文</returns>
private static byte[] GetArrayDataWriteMessage(int slaveStation, short startAdr, IEnumerable<short> value)
{
//定义报文临时存储字节集合
List<byte> tempList = new List<byte>();
//插入从站地址
tempList.Add((byte)slaveStation);
//插入功能码
tempList.Add(0x10);
//获取起始地址
byte[] start = BitConverter.GetBytes(startAdr);
//获取写入值的数量
byte[] length = BitConverter.GetBytes(Convert.ToInt16(value.Count()));
//根据计算机大小端存储方式进行高低字节转换
if (BitConverter.IsLittleEndian)
{
Array.Reverse(start);
Array.Reverse(length);
}
//插入起始地址
tempList.AddRange(start);
//插入写入值数量
tempList.AddRange(length);
//创建写入值字节集合
List<byte> valueBytes = new List<byte>();
//将需要插入的每个值转换为字节数组,
//并根据计算机大小端存储方式进行高低字节转换
//然后插入到值的字节集合中
foreach (var item in value)
{
byte[] temp = BitConverter.GetBytes(item);
if (BitConverter.IsLittleEndian)
{
Array.Reverse(temp);
}
valueBytes.AddRange(temp);
}
//获取写入值的字节数
byte count = Convert.ToByte(valueBytes.Count);
//插入写入值的字节数
tempList.Add(count);
//插入写入值字节集合
tempList.AddRange(valueBytes);
//根据报文字节集合计算CRC16校验码,并拼接到最后,然后转换为字节数组并返回
return tempList.Concat(CheckSum.CRC16(tempList)).ToArray();
}
/// <summary>
/// 反转顺序并生成字节
/// </summary>
/// <param name="data">位数据</param>
/// <returns></returns>
private static byte GetBitArray(IEnumerable<bool> data)
{
//把位数据集合反转
data.Reverse();
//定义初始字节,值为0000 0000
byte temp = 0x00;
//循环计数
int index = 0;
//循环位集合
foreach (bool item in data)
{
//判断每一位的数据,为true则左移一个1到对应的位置
if (item) temp = (byte)(temp | (0x01 << index));
//计数+1
index++;
}
//返回最后使用位数据集合生成的二进制字节
return temp;
}
}
这里需要注意,我们所有生成报文的方法,都是通过MessageGeneration这个方法进行调用,这样在使用时,我们只需要传入不同的参数,就可以实现对不同方法的调用,并根据读写模式的不同使用不同的重载方法。而具体生成方法需要修改的时候,基本不需要修改已经调用了MessageGeneration方法的代码。
最后是解析接收从站响应报文的方法,此处不添加浮点数等数据类型的解析。如果需要浮点数处理,可以参考前面的读取报文生成的文章。
AnalysisMessage:
class AnalysisMessage
{
/// <summary>
/// 解析线圈数据
/// </summary>
/// <param name="receiveMsg">接收到的报文</param>
/// <returns></returns>
public static BitArray GetCoil(byte[] receiveMsg)
{
//获取线圈状态
BitArray bitArray = new BitArray(receiveMsg.Skip(3).Take(Convert.ToInt32(receiveMsg[2])).ToArray());
return bitArray;
}
/// <summary>
/// 解析寄存器数据
/// </summary>
/// <param name="receiveMsg">接收到的报文</param>
/// <returns></returns>
public static List<short> GetRegister(byte[] receiveMsg)
{
List<short> result = new List<short>();
//获取字节数
int count = Convert.ToInt32(receiveMsg[2]);
int index = 0;
for (int i = 3; i < count + 3; i += 2)
{
index++;
//每个地址所属的字节数组
byte[] temp = new byte[] { receiveMsg[i + 1], receiveMsg[i] };
//获取整型结果
result.Add(BitConverter.ToInt16(temp, 0));
}
return result;
}
}
窗体
通讯建立
窗体通过按钮点击,打开串口连接。
/// <summary>
/// 打开或者关闭串口连接
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnCon_Click(object sender, EventArgs e)
{
if (!conHelper.Status)
{
//串口号
string port = cbxPort.SelectedItem.ToString();
//波特率
int baudrate = (int)cbxBaudRate.SelectedItem;
//奇偶校验
Parity parity = GetSelectedParity();
//数据位
int databits = (int)cbxDataBits.SelectedItem;
//停止位
StopBits stopBits = GetSelectedStopBits();
//设定串口参数
conHelper.SetSerialPort(port, baudrate, parity, databits, stopBits);
//打开串口
conHelper.Open();
Thread.Sleep(200);
//刷新状态
tbxStatus.Text = conHelper.Status ? "连接成功" : "未连接";
//启用读写按钮
btnRW.Enabled = true;
btnCon.Text = "关闭串口";
}
else
{
//关闭串口
conHelper.Close();
tbxStatus.Text = conHelper.Status ? "连接成功" : "未连接";
//禁用读写按钮
btnRW.Enabled = false;
btnCon.Text = "打开串口";
}
}
这里其实就是简单的打开和关闭串口的连接,如果串口未打开,则禁用读写按钮。
模式切换
根据读写模式的下拉列表的不同,修改窗体的存储读写模式的字段的值。
/// <summary>
/// 读写模式切换事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void cbxMode_SelectedIndexChanged(object sender, EventArgs e)
{
//更新状态字段
GetReadWriteMode();
//计数复位
nudCount.Value = 1;
//清空输入值
tbxValue.Clear();
//是否显示提示文本
labTip.Visible = isSingleData ? false : true;
//是否可输入值
tbxValue.Enabled = isWrite ? true : false;
//是否可修改计数
nudCount.Enabled = isWrite ? false : true;
//读写按钮显示文本
btnRW.Text = isWrite ? "写入" : "读取";
}
读写事件
/// <summary>
/// 读写按钮事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnRW_Click(object sender, EventArgs e)
{
//生成的报文
byte[] message = null;
//从站地址
short station = (short)nudStation.Value;
//起始地址
short stratAdr = (short)nudAddress.Value;
//读写数量
short count = (short)nudCount.Value;
if (isWrite)
{
//生成写入报文
WriteMode mode = (WriteMode)readWriteMode;
//生成单个或多个值的写入报文
if (isSingleData)
{
//判断是否输入单个值
if (tbxValue.Text.IndexOf(",") != -1)
{
MessageBox.Show("输入值过多");
return;
}
//生成写入单个值的写入报文
if (isCoil)
{
//生成写入单个线圈的报文
bool value = false;
if (string.Equals(tbxValue.Text, "True", StringComparison.OrdinalIgnoreCase) || tbxValue.Text == "1")
{
value = true;
}
else if (string.Equals(tbxValue.Text, "False", StringComparison.OrdinalIgnoreCase) || tbxValue.Text == "0")
{
value = false;
}
else
{
MessageBox.Show("输入值只能是1、0或者true、false");
return;
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
else
{
//生成写入单个寄存器的报文
try
{
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, short.Parse(tbxValue.Text));
}
catch (Exception)
{
MessageBox.Show("输入有误");
return;
}
}
}
else
{
//输入值数组
string[] arr = tbxValue.Text.Split(",");
if (isCoil)
{
//生成写入多个线圈的报文
List<bool> value = new List<bool>();
for (int i = 0; i < arr.Length; i++)
{
bool temp = false;
if (string.Equals(arr[i], "True", StringComparison.OrdinalIgnoreCase) || arr[i] == "1")
{
temp = true;
}
else if (string.Equals(tbxValue.Text, "False", StringComparison.OrdinalIgnoreCase) || arr[i] == "0")
{
temp = false;
}
else
{
MessageBox.Show("输入值只能是1、0或者true、false");
return;
}
value.Add(temp);
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
else
{
//生成写入多个寄存器的报文
try
{
List<short> value = new List<short>();
for (int i = 0; i < arr.Length; i++)
{
value.Add(short.Parse(arr[i]));
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
catch (Exception)
{
MessageBox.Show("输入有误");
return;
}
}
}
}
else
{
//生成读取报文
ReadMode mode = (ReadMode)readWriteMode;
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, count);
}
//发送报文
conHelper.SendDataMethod(message);
//将发送的报文显示在窗体中
string msgStr = "";
for (int i = 0; i < message.Length; i++)
{
msgStr += message[i].ToString("X2") + " ";
}
rbxSendMsg.Text = msgStr;
}
根据选择的模式不一样,以及输入的值的不同,我们做一次初步的解析,生成对应的写入值或者调用对应的读取方法,来获取读写报文,并通过串口发送出去,然后在窗口中显示我们所发送的报文。
消息接收事件
/// <summary>
/// 接收消息的事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e">事件参数</param>
private void ReceiveDataEvent(object sender, ReceiveDataEventArg e)
{
//在窗体上显示接收到的报文
string msgStr = "";
for (int i = 0; i < e.Data.Length; i++)
{
msgStr += e.Data[i].ToString("X2") + " ";
}
rbxRecMsg.Invoke(new Action(() => { rbxRecMsg.Text = msgStr; }));
//如果是读取数据,则对接收到的消息进行解析
if (!isWrite)
{
string result = "";
if (isCoil)
{
BitArray bitArray = AnalysisMessage.GetCoil(e.Data);
int count = Convert.ToInt32(nudCount.Value);
for (int i = 0; i < count; i++)
{
result += bitArray[i].ToString() + ",";
}
}
else
{
List<short> list = AnalysisMessage.GetRegister(e.Data);
list.ForEach(m => { result += m.ToString() + ","; });
}
tbxValue.Invoke(new Action(() => { tbxValue.Text = result.Remove(result.LastIndexOf(","), 1); }));
}
}
窗体完整代码
using ModbusRTUDemo.Communication;
using ModbusRTUDemo.Message;
using System;
using System.Collections;
using System.Collections.Generic;
using System.IO.Ports;
using System.Text.RegularExpressions;
using System.Threading;
using System.Windows.Forms;
namespace ModbusRTUDemo
{
public partial class DemoForm : Form
{
#region Field
/// <summary>
/// 串口类
/// </summary>
SerialPortHelper conHelper = new SerialPortHelper();
/// <summary>
/// 是否为写入模式
/// </summary>
private bool isWrite = false;
/// <summary>
/// 是否读写线圈
/// </summary>
private bool isCoil = true;
/// <summary>
/// 是否读写单个值
/// </summary>
private bool isSingleData = true;
/// <summary>
/// 读写模式
/// </summary>
private object readWriteMode = null;
#endregion
#region Ctor
public DemoForm()
{
InitializeComponent();
}
#endregion
#region FormEvent
/// <summary>
/// 窗体加载事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void DemoForm_Load(object sender, EventArgs e)
{
//设置可选串口
cbxPort.Items.AddRange(SerialPortHelper.GetPortArray());
//设置可选波特率
cbxBaudRate.Items.AddRange(new object[] { 9600, 19200 });
//设置可选奇偶校验
cbxParity.Items.AddRange(new object[] { "None", "Odd", "Even", "Mark", "Space" });
//设置可选数据位
cbxDataBits.Items.AddRange(new object[] { 5, 6, 7, 8 });
//设置可选停止位
cbxStopBits.Items.AddRange(new object[] { 1, 1.5, 2 });
//设置读写模式
cbxMode.Items.AddRange(new object[] {
"读取输出线圈",
"读取离散输入",
"读取保持型寄存器",
"读取输入寄存器",
"写入单个线圈",
"写入多个线圈",
"写入单个寄存器",
"写入多个寄存器"
});
//设置默认选中项
cbxPort.SelectedIndex = 1;
cbxBaudRate.SelectedIndex = 0;
cbxParity.SelectedIndex = 0;
cbxDataBits.SelectedIndex = 3;
cbxStopBits.SelectedIndex = 0;
cbxMode.SelectedIndex = 0;
//显示连接状态
tbxStatus.Text = conHelper.Status ? "连接成功" : "未连接";
//从站地址默认为1
nudStation.Value = 1;
//设置为默认输入法,即为英文半角
tbxValue.ImeMode = ImeMode.Disable;
//初始化禁用读写按钮(未打开串口连接)
btnRW.Enabled = false;
//注册接收消息的事件
conHelper.ReceiveDataEvent += ReceiveDataEvent;
}
/// <summary>
/// 打开或者关闭串口连接
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnCon_Click(object sender, EventArgs e)
{
if (!conHelper.Status)
{
//串口号
string port = cbxPort.SelectedItem.ToString();
//波特率
int baudrate = (int)cbxBaudRate.SelectedItem;
//奇偶校验
Parity parity = GetSelectedParity();
//数据位
int databits = (int)cbxDataBits.SelectedItem;
//停止位
StopBits stopBits = GetSelectedStopBits();
//设定串口参数
conHelper.SetSerialPort(port, baudrate, parity, databits, stopBits);
//打开串口
conHelper.Open();
Thread.Sleep(200);
//刷新状态
tbxStatus.Text = conHelper.Status ? "连接成功" : "未连接";
//启用读写按钮
btnRW.Enabled = true;
btnCon.Text = "关闭串口";
}
else
{
//关闭串口
conHelper.Close();
tbxStatus.Text = conHelper.Status ? "连接成功" : "未连接";
//禁用读写按钮
btnRW.Enabled = false;
btnCon.Text = "打开串口";
}
}
/// <summary>
/// 读写按钮事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnRW_Click(object sender, EventArgs e)
{
//生成的报文
byte[] message = null;
//从站地址
short station = (short)nudStation.Value;
//起始地址
short stratAdr = (short)nudAddress.Value;
//读写数量
short count = (short)nudCount.Value;
if (isWrite)
{
//生成写入报文
WriteMode mode = (WriteMode)readWriteMode;
//生成单个或多个值的写入报文
if (isSingleData)
{
//判断是否输入单个值
if (tbxValue.Text.IndexOf(",") != -1)
{
MessageBox.Show("输入值过多");
return;
}
//生成写入单个值的写入报文
if (isCoil)
{
//生成写入单个线圈的报文
bool value = false;
if (string.Equals(tbxValue.Text, "True", StringComparison.OrdinalIgnoreCase) || tbxValue.Text == "1")
{
value = true;
}
else if (string.Equals(tbxValue.Text, "False", StringComparison.OrdinalIgnoreCase) || tbxValue.Text == "0")
{
value = false;
}
else
{
MessageBox.Show("输入值只能是1、0或者true、false");
return;
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
else
{
//生成写入单个寄存器的报文
try
{
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, short.Parse(tbxValue.Text));
}
catch (Exception)
{
MessageBox.Show("输入有误");
return;
}
}
}
else
{
//输入值数组
string[] arr = tbxValue.Text.Split(",");
if (isCoil)
{
//生成写入多个线圈的报文
List<bool> value = new List<bool>();
for (int i = 0; i < arr.Length; i++)
{
bool temp = false;
if (string.Equals(arr[i], "True", StringComparison.OrdinalIgnoreCase) || arr[i] == "1")
{
temp = true;
}
else if (string.Equals(tbxValue.Text, "False", StringComparison.OrdinalIgnoreCase) || arr[i] == "0")
{
temp = false;
}
else
{
MessageBox.Show("输入值只能是1、0或者true、false");
return;
}
value.Add(temp);
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
else
{
//生成写入多个寄存器的报文
try
{
List<short> value = new List<short>();
for (int i = 0; i < arr.Length; i++)
{
value.Add(short.Parse(arr[i]));
}
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, value);
}
catch (Exception)
{
MessageBox.Show("输入有误");
return;
}
}
}
}
else
{
//生成读取报文
ReadMode mode = (ReadMode)readWriteMode;
message = MessageGenerationModule.MessageGeneration(station, mode, stratAdr, count);
}
//发送报文
conHelper.SendDataMethod(message);
//将发送的报文显示在窗体中
string msgStr = "";
for (int i = 0; i < message.Length; i++)
{
msgStr += message[i].ToString("X2") + " ";
}
rbxSendMsg.Text = msgStr;
}
/// <summary>
/// 读写模式切换事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void cbxMode_SelectedIndexChanged(object sender, EventArgs e)
{
//更新状态字段
GetReadWriteMode();
//计数复位
nudCount.Value = 1;
//清空输入值
tbxValue.Clear();
//是否显示提示文本
labTip.Visible = isSingleData ? false : true;
//是否可输入值
tbxValue.Enabled = isWrite ? true : false;
//是否可修改计数
nudCount.Enabled = isWrite ? false : true;
//读写按钮显示文本
btnRW.Text = isWrite ? "写入" : "读取";
}
/// <summary>
/// 根据输入值的数量同步刷新窗体计数
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void tbxValue_TextChanged(object sender, EventArgs e)
{
nudCount.Value = Regex.Matches(tbxValue.Text, ",").Count + 1;
}
#endregion
#region Event
/// <summary>
/// 接收消息的事件
/// </summary>
/// <param name="sender"></param>
/// <param name="e">事件参数</param>
private void ReceiveDataEvent(object sender, ReceiveDataEventArg e)
{
//在窗体上显示接收到的报文
string msgStr = "";
for (int i = 0; i < e.Data.Length; i++)
{
msgStr += e.Data[i].ToString("X2") + " ";
}
rbxRecMsg.Invoke(new Action(() => { rbxRecMsg.Text = msgStr; }));
//如果是读取数据,则对接收到的消息进行解析
if (!isWrite)
{
string result = "";
if (isCoil)
{
BitArray bitArray = AnalysisMessage.GetCoil(e.Data);
int count = Convert.ToInt32(nudCount.Value);
for (int i = 0; i < count; i++)
{
result += bitArray[i].ToString() + ",";
}
}
else
{
List<short> list = AnalysisMessage.GetRegister(e.Data);
list.ForEach(m => { result += m.ToString() + ","; });
}
tbxValue.Invoke(new Action(() => { tbxValue.Text = result.Remove(result.LastIndexOf(","), 1); }));
}
}
#endregion
#region Methods
/// <summary>
/// 获取窗体选中的奇偶校验
/// </summary>
/// <returns></returns>
private Parity GetSelectedParity()
{
switch (cbxParity.SelectedItem.ToString())
{
case "Odd":
return Parity.Odd;
case "Even":
return Parity.Even;
case "Mark":
return Parity.Mark;
case "Space":
return Parity.Space;
case "None":
default:
return Parity.None;
}
}
/// <summary>
/// 获取窗体选中的停止位
/// </summary>
/// <returns></returns>
private StopBits GetSelectedStopBits()
{
switch (Convert.ToDouble(cbxStopBits.SelectedItem))
{
case 1:
return StopBits.One;
case 1.5:
return StopBits.OnePointFive;
case 2:
return StopBits.Two;
default:
return StopBits.One;
}
}
/// <summary>
/// 根据选中的读写模式更新字段值
/// </summary>
private void GetReadWriteMode()
{
switch (cbxMode.SelectedItem.ToString())
{
case "读取输出线圈":
default:
isWrite = false;
isSingleData = false;
isCoil = true;
readWriteMode = ReadMode.Read01;
break;
case "读取离散输入":
isWrite = false;
isSingleData = false;
isCoil = true;
readWriteMode = ReadMode.Read02;
break;
case "读取保持型寄存器":
isWrite = false;
isSingleData = false;
isCoil = false;
readWriteMode = ReadMode.Read03;
break;
case "读取输入寄存器":
isWrite = false;
isSingleData = false;
isCoil = false;
readWriteMode = ReadMode.Read04;
break;
case "写入单个线圈":
isWrite = true;
isSingleData = true;
isCoil = true;
readWriteMode = WriteMode.Write01;
break;
case "写入多个线圈":
isWrite = true;
isSingleData = false;
isCoil = true;
readWriteMode = WriteMode.Write01s;
break;
case "写入单个寄存器":
isWrite = true;
isSingleData = true;
isCoil = false;
readWriteMode = WriteMode.Write03;
break;
case "写入多个寄存器":
isWrite = true;
isSingleData = false;
isCoil = false;
readWriteMode = WriteMode.Write03s;
break;
}
}
#endregion
}
}
测试
读取线圈
读取离散输入
读取保持型寄存器
读取输入寄存器
写入单个线圈
写入多个线圈
写入单个寄存器
写入多个寄存器
结尾
至此,我们的ModbusRTU的Winform的Demo就已经完成了,可以看到,八种报文的测试都是没有问题的。在实际使用的时候,我们可以选择像本文介绍的方法,自己创建连接并生成发送报文,也可以使用现有的类库,下一篇将会介绍一个现有的ModbusRTU的类库的使用。
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)