Modbus协议在应用中一般用来与PLC或者其他硬件设备通讯,Modbus集成到IoTBrowser使用串口插件模式开发,不同的是采用命令函数,具体可以参考前面几篇文章。目前示例实现了Modbus-Rtu和Modbus-Tcp两种,通过js可以与Modbus进行通讯控制。
一、开发插件
- 添加引用
- 添加NModbus4,在NuGet搜索NModbus4
- 添加Core,路径:\IoTBrowser\src\app_x64\Core.dll
- 添加Infrastructure,路径:\IoTBrowser\src\app_x64\Infrastructure.dll
- 添加Newtonsoft,路径:\IoTBrowser\src\app_x64\Newtonsoft.Json.dll
- 开发ModbusRtu和ModbusTcp插件
- ModbusRtu
- 添加引用
public class ModbusRtuCom : ComBase { public override string Type => "modbusRtuCom"; public override string Name => "ModbusRtuCom"; private object _locker = new object(); public override bool Init(int port, int baudRate = 9600, string extendData = null) { this.Port = port; var portName = "COM" + port; base.PortName = portName; ModbusRtuService.Init(portName, baudRate); Console.WriteLine("初始化ModbusRtuCom驱动程序成功!"); return true; } public override event PushData OnPushData; public override bool Open() { var b = false; try { ModbusRtuService.Open(); b = true; IsOpen = true; } catch (Exception ex) { string msg = string.Format("ModbusRtuCom串口打开失败:{0} ", ex.Message); Console.WriteLine(msg); } return b; } public override bool Close() { ModbusRtuService.Close(); IsOpen = false; OnPushData = null; return true; } public override string Command(string name, string data) { var outData = string.Empty; var dataObj = Newtonsoft.Json.JsonConvert.DeserializeObject<dynamic>(data); switch (name) { case "ReadCoils": //01 var readData = ModbusRtuService.ReadCoils(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadInputs": //02 readData = ModbusRtuService.ReadInputs(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadHoldingRegisters": //03 readData = ModbusRtuService.ReadHoldingRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadInputRegisters": //04 readData = ModbusRtuService.ReadInputRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "WriteSingleCoil": //05 ModbusRtuService.WriteSingleCoil(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ModbusHelper.BoolParse(dataObj.value.ToString())); break; case "WriteSingleRegister": //06 ModbusRtuService.WriteSingleRegister(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.value.ToString())); break; case "WriteMultipleCoils": //0F 写一组线圈 var values = dataObj.value.ToString().Split(' '); var datas = new bool[values.Length]; for (var i = 0; i < values.Length; i++) { datas[i] = ModbusHelper.BoolParse(values[i]); } ModbusRtuService.WriteMultipleCoils(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), datas); break; case "WriteMultipleRegisters": // 10 写一组保持寄存器 values = dataObj.value.ToString().Split(' '); var udatas = new ushort[values.Length]; for (var i = 0; i < values.Length; i++) { udatas[i] = ushort.Parse(values[i]); } ModbusRtuService.WriteMultipleRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), udatas); break; } return outData; } }
b.ModbusTcp
public class ModbusTcpCom : ComBase { public override string Type => "modbusTcpCom"; public override string Name => "ModbusTcpCom"; private object _locker = new object(); public override bool Init(int port, int baudRate = 9600, string extendData = null) { this.Port = port; ModbusTcpService.Init(extendData, port); Console.WriteLine("初始化ModbusTcpCom驱动程序成功!"); return true; } public override event PushData OnPushData; public override bool Open() { var b = false; try { ModbusTcpService.Open(); b = true; IsOpen = true; } catch (Exception ex) { string msg = string.Format("ModbusTcpCom串口打开失败:{0} ", ex.Message); Console.WriteLine(msg); } return b; } public override bool Close() { ModbusTcpService.Close(); IsOpen = false; OnPushData = null; return true; } public override string Command(string name, string data) { var outData = string.Empty; var dataObj = Newtonsoft.Json.JsonConvert.DeserializeObject<dynamic>(data); switch (name) { case "ReadCoils": //01 var readData = ModbusTcpService.ReadCoils(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadInputs": //02 readData = ModbusTcpService.ReadInputs(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadHoldingRegisters": //03 readData = ModbusTcpService.ReadHoldingRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "ReadInputRegisters": //04 readData=ModbusTcpService.ReadInputRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.numberOfPoints.ToString())); outData = ModbusHelper.ToString(readData); break; case "WriteSingleCoil": //05 ModbusTcpService.WriteSingleCoil(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ModbusHelper.BoolParse(dataObj.value.ToString())); break; case "WriteSingleRegister": //06 ModbusTcpService.WriteSingleRegister(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), ushort.Parse(dataObj.value.ToString())); break; case "WriteMultipleCoils": //0F 写一组线圈 var values = dataObj.value.ToString().Split(' '); var datas =new bool[values.Length]; for(var i=0;i< values.Length;i++) { datas[i] = ModbusHelper.BoolParse(values[i]); } ModbusTcpService.WriteMultipleCoils(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), datas); break; case "WriteMultipleRegisters": // 10 写一组保持寄存器 values = dataObj.value.ToString().Split(' '); var udatas = new ushort[values.Length]; for (var i = 0; i < values.Length; i++) { udatas[i] = ushort.Parse(values[i]); } ModbusTcpService.WriteMultipleRegisters(byte.Parse(dataObj.slaveAddress.ToString()), ushort.Parse(dataObj.startAddress.ToString()), udatas); break; } return outData; } }
3.功能
- 读单个线圈
- 读取输入线圈/离散量线圈
- 读取保持寄存器
- 读取输入寄存器
- 写单个线圈
- 写单个输入线圈/离散量线圈
- 写一组线圈
- 写一组保持寄存器
源代码位置:\Plugins\DDS.IoT.Modbus
二、本机测试
1.测试前准备
需要安装虚拟串口和modbusslave,可以在源代码中下载:
https://gitee.com/yizhuqing/IoTBrowser/tree/master/soft
2.串口测试
3.TCP测试
三、部署到IoTBrowser
1.编译
(建议生产环境使用Release模式)
2.拷贝到Plugins文件夹
也可以放到com文件夹。
注意:需要拷贝NModbus4.dll到\IoTBrowser\src\app_x64目录下
四、IoTBrowser集成测试
1.串口测试
写入多个数据写入以空格分割,写入线圈数据支持0/1或false/true。
2.TCP测试
TCP注意ip地址通过扩展数据传入,端口号就是串口号。