C#使用NModbus4库创建Modbus TCP Slave(服务器)简单示例

服务器 0

本文续上篇Codesys—标准库ModbusTCP Master(客户端)配合C#的NModbus4库的通讯示例

链接:https://blog.csdn.net/wushangwei2019/article/details/136375234?spm=1001.2014.3001.5501

上篇描述在Codesys端的Modbus TCP Master(客户端)的设备添加、IO映射、通讯简单展示等方面,本文记录PC端C#利用NModbus4通讯库创建Modbus TCP Slave(服务器)的方法。

注:本文只记录如何使用NModbus4的部分功能,程序结构较为简单,并不适用于项目工程。

本文分以下几个步骤分享NModbus4的使用:

1.添加NModbus4库

2.ModbusTcpSlave从站的创建

3.事件订阅

4.通讯示例

界面附加显示功能:

1.在收到客户端报文的事件后,将会在信息提示框中显示报文内容,报文内容不包含CRC;在服务器写入完成后,将会在信息提示框中显示写入完成的报文内容。

2.在服务器线程中,获取已连接的客户端的IP地址以及端口号信息,并在最底下显示出来。

软件界面如下,本文针对左侧服务器(从站)端:

1.添加NModbus4库 

菜单栏点击【工具】-》【NuGet包管理器】-》【管理解决方案的NuGet程序包】

弹出以下画面,搜索【NModbus4】 ,选择需要安装的项目,并点击右下角的【安装】即可。

在程序中添加如下代码:

using Modbus.Data;using Modbus.Device;using Modbus.Extensions.Enron;

 2.使用TcpListener创建服务器

核心代码:

listener = new TcpListener(IPAddress.Parse(textBox2.Text), Convert.ToInt32(textBox3.Text));
listener.Start();
slave = ModbusTcpSlave.CreateTcp(1, listener);
slave.DataStore = DataStoreFactory.CreateDefaultDataStore(); 

slave.Listen();

解析:

1.TcpListener,用于创建服务器,需要输入参数待创建服务器的【IP地址】和【端口号】。

2.ModbusTcpSlave.CreateTcp方法,用于创建ModbusTCP Slave从站,创建后的从站对象为slave。

3.DataStoreFactory.CreateDefaultDataStore()方法,用于清除Modbus TCP Slave的数据存储区,寄存器区值全部写0。

4.slave.Listen()方法,Modbus TCP Slave开始监听请求,我的理解是有客户端连接后,Slave开始响应客户端的报文,此方法应该放在有客户端连接后再使用较好,但在此处调用也能正常运行。

            try            {                listener = new TcpListener(IPAddress.Parse(textBox2.Text), Convert.ToInt32(textBox3.Text));                listener.Start();                slave = ModbusTcpSlave.CreateTcp(1, listener);                slave.DataStore = DataStoreFactory.CreateDefaultDataStore();                WriteInfo("创建服务器成功!" + "线程ID:" + Thread.CurrentThread.ManagedThreadId + "/r/n");                //订阅数据到达事件,不能获取具体接收到的报文                //slave.DataStore.DataStoreWrittenTo += DataStoreWrittenToHandle;                //订阅接收到报文请求事件,可以打印接收到的报文                slave.ModbusSlaveRequestReceived += ModbusSlaveRequestReceivedHandle;                //订阅接收到写入完成事件,可以打印写入完成响应                slave.WriteComplete += WriteCompleteHandle;                slave.Listen();                isServerCreated = true;                WriteInfo("服务器创建成功!" +  "/r/n");                count2 = 0;            }            catch(Exception ex)            {                count2++;                 WriteInfo("创建服务器失败!" + "失败次数:"+count2.ToString()+ "/r/n");                isServerCreated = false;                creatServer= false;                 return;            }

3.事件订阅

分别说明以下三个事件的功能。

//订阅数据到达事件,不能获取具体接收到的报文
//slave.DataStore.DataStoreWrittenTo += DataStoreWrittenToHandle;
//订阅接收到报文请求事件,可以打印接收到的报文
slave.ModbusSlaveRequestReceived += ModbusSlaveRequestReceivedHandle;
 //订阅接收到写入完成事件,可以打印写入完成响应
slave.WriteComplete += WriteCompleteHandle;

1.DataStore.DataStoreWrittenTo事件:当DataStore通过Modbus命令被写入数据时触发。

2.ModbusSlaveRequestReceived事件:当Slave收到主站报文时触发。

3.WriteComplete事件:当Slave接收到主站报文,并写入完成后触发。

本例主要使用后面【ModbusSlaveRequestReceived事件】和【WriteComplete事件】。

ModbusSlaveRequestReceived事件触发时,调用函数在提示框中打印报文信息,报文信息转换成16进制显示。

        private void ModbusSlaveRequestReceivedHandle(object obj, ModbusSlaveRequestEventArgs e)        {            string str = "";            foreach (var item in e.Message.MessageFrame)            {                str += item.ToString("x2").PadLeft(2, '0').ToUpper() + "  ";                }            WriteInfo("服务器收到报文:  " + str + "/r/n");        }

WriteComplete事件,调用函数在提示框中打印报文信息,报文信息转换成16进制显示。

        private void WriteCompleteHandle(object sender, ModbusSlaveRequestEventArgs e)        {            string str = "";            foreach (var item in e.Message.MessageFrame)            {                str += item.ToString("x2").PadLeft(2, '0').ToUpper() + "  ";            }            WriteInfo("服务器写入完成:  " + str + "/r/n");        }

4.通讯示例

在服务器侧画面上,输入【IP】:127.0.0.1,【Port】:502,点击【创建服务器】,可观察到下方提示框中显示【创建服务器成功】。

打开Modbus Poll软件,在连接设置在远端服务器侧输入【IP地址】和【端口号】,点击确认连接。

连接成功后,PC端软件提示服务器收到报文,报文信息没有包含CRC字节信息;Modbus Poll端软件显示Tx=11,不断增加,这是由于默认使用了03功能读保持寄存器,数量长度为20个字,并循环读取。

Modbus Poll的配置如下: 

根据PC端软件已有功能,进行测试。

从站(服务器)写入,Modbus Poll(主站/客户端)读取:

【Value】输入2,点击【写1~10】,将向保持寄存器地址【1~10】写入值为Value*地址值,如:地址1=2;

地址2=4;

地址3=6;

地址4=8;

依次类推...如下图所示。

Modbus Poll(主站/客户端)写入,从站(服务器)读取:

Modbus Poll使用03功能码往保持寄存器地址11,写入123。 左侧软件提示框显示“服务器收到报文+收到报文”,此提示是由【ModbusSlaveRequestReceived事件】触发后发出;然后显示“服务器写入完成+收到报文”,此提示是由【WriteComplete事件】触发后发出。

Modbus Poll使用16功能码往保持寄存器地址11开始写10个寄存器,命令配置如下:

观察左侧软件提示框信息:

报文数据中的  00  0B  00  0C  00  0D  00  0E  00  0F  00  10  00  11  00  12  00  13  00  14

转换成10进制为11(00 0B),12(00 0C),13,14,15,16,17,18,19,20。

至此,基本通讯测试完成。

下面补充服务器端,如何对保持寄存器进行读写,其实很简单,就是直接访问slave.DataStore.HoldingRegisters[i]中的数据即可。

如下为写入代码:

            if (listener != null && slave!=null) {                for (int i = 1;i<11;i++)                {                    slave.DataStore.HoldingRegisters[i] = (ushort)(numericUpDown2 .Value* i);                }                WriteInfo("服务器写入1~10完成" + "/r/n");            }

以下为读取代码:

            if(slave!=null)            {                string str = "";                for(int i=10;i<20;i++)                {                    str += slave.DataStore.HoldingRegisters[i+1].ToString() + "  ";                }                WriteInfo("读取寄存器(地址为10~20):"+str +"/r/n");             }

也许您对下面的内容还感兴趣: