C#使用NModbus4库创建Modbus TCP Slave(服务器)简单示例
创始人
2025-01-20 21:03:56
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");              }

相关内容

热门资讯

ArcGIS无法链接在线地图或... 这几天我们分享了!谷歌卫星影像图归来!ArcGIS直连!快...
《详细》使用ventoy制作l... 通过docker部署一键爬虫,爬取海量电影;部署ai绘画;...
Vue 大文件切片上传实现指南... Vue 大文件切片上传实现指南背景        在Web开发中,文件上传是一个常见的...
解决Pycharm运行服务器文... 解决Pycharm运行服务器文件时出现Cannot find remote credentials ...
解决服务器能ping通但端口不... 前言是本地机房做的esxi系统,测试环境使用的虚拟机。一切都哼正常,突然...
IDEA2023连接服务器do... 文章目录TCP 方式连接docker1. 服务器docker配置修改查看虚拟机中Docker配置文件...
idea实现ssh远程连接服务... 1. 首先,打开idea,点击左上角File->settings2. 点...
ubuntu搭建freerad... 1. 综述802.1X 是一种基于端口的网络访问控制协议,常用于无线局域网࿰...
【Linux】十分钟快速了解L... 目录💖一. 关机指令01. shutdown02. halt03. reboot&#...
服务器遭遇挖矿病毒syst3m... 01 什么是挖矿病毒挖矿病毒通常是恶意软件的一种,它会在受感染的系统上无授权地挖掘加密...