物联网浏览器(IoTBrowser)-Modbus协议集成和测试
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地址通过扩展数据传入,端口号就是串口号。
物联网浏览器(IoTBrowser)-Modbus协议集成和测试的更多相关文章
- Modbus协议和应用开发介绍
因业务需要了解Modbus协议的使用,因此对Modbus的协议,以及相应的C#处理应用进行了解,针对协议的几种方式(RTU.ASCII.TCPIP)进行了封装,以及对Modbus的各种功能码的特点进行 ...
- 《ServerSuperIO Designer IDE使用教程》-3.Modbus协议,读取多个寄存器,实现多种数据类型解析。发布:v4.2.2版本
更新内容,v4.2.2版本:1.增加Modbus协议读取多个寄存器,并且按多种数据类型解析数据.2.Modbus Serial和Modbus TCP两个驱动合并成一个驱动.3.修改数据库结构,保存配置 ...
- Modbus通信协议的压力测试
最近物联网都比较的火,因此,特别为各位兄弟姐妹们,奉上一款Mobus协议的测试软件,可以用来做设备的压力测试,和通信测试. 起初软件开发缘由是我们最近在开发一款设备,需要将多个DS18B20并联起来, ...
- MQTT是IBM开发的一个即时通讯协议,构建于TCP/IP协议上,是物联网IoT的订阅协议,借助消息推送功能,可以更好地实现远程控制
最近一直做物联网方面的开发,以下内容关于使用MQTT过程中遇到问题的记录以及需要掌握的机制原理,主要讲解理论. 背景 MQTT是IBM开发的一个即时通讯协议.MQTT构建于TCP/IP协议上,面向M2 ...
- Modbus 协议-Ascii,RTU
Modbus 协议之 Ascii 下载地址:http://download.csdn.net/detail/woxpp/5043249 1.提供Modbus Ascii 相关发送与接收代码 2.提供M ...
- Modbus协议深入讲解_NI
from:https://www.ni.com/zh-cn/innovations/white-papers/14/the-modbus-protocol-in-depth.html 已更新 Mar ...
- 将jacoco集成到测试工具平台
最近在做接口测试,想通过代码覆盖率来判断一下接口用例是否缺失,但是每次通过命令来生成覆盖率报告,感觉太麻烦,所以就想着把jacoco集成到测试工具平台中,只需要点几个按钮,就能查看到覆盖率报告. 测试 ...
- 基于奇林软件kylinTOP工具的HTTP2协议的压力测试
1.HTTP协议概述 说到http,那就应该先了解一下http协议的发展历史.关于http协议的历史,可以参考阮一峰老师的这篇博客文章HTTP 协议入门,里面介绍的比较详细了.简单来说http先后存在 ...
- Modbus协议及python库实现
基础知识 硬件层协议:解决0和1的可靠传输,常有RS232.RS485.CAN.IIC.SPI - 软件层协议:解决传输目的,常有Modbus.TCP/IP.CANopen - 协议优点: Modbu ...
- modbus协议讲义
Modbus 一个工业上常用的通讯协议.一种通讯约定.Modbus协议包括RTU.ASCII.TCP.其中MODBUS-RTU最常用,比较简单,在单片机上很容易实现.虽然RTU比较简单,但是看 ...
随机推荐
- Leaflet 地图偏移 地图纠偏
(地图瓦片纠偏最好的方法在这:https://www.cnblogs.com/s0611163/p/15606460.html) 地图区域是一个市,偏移量可以近似认为是固定不变的,通过修改Leafle ...
- Python 使用pip或easy_install或手动安装库/模块/软件包
pip是easy_install的改进版,提供更好的提示信息,删除package等功能.老版本的python中只有easy_install,没有pip.本文主要介绍使用pip或easy_install ...
- LeetCode | 983.最低票价(动态规划)
在一个火车旅行很受欢迎的国度,你提前一年计划了一些火车旅行.在接下来的一年里,你要旅行的日子将以一个名为days的数组给出.每一项是一个从 1 到 365 的整数. 火车票有三种不同的销售方式: 一张 ...
- 活动回顾|阿里云 Serverless 技术实战与创新广州站回放&PPT下载
7月8日"阿里云 Serverless 技术实战与创新"广州站圆满落幕.活动受众以关注Serverless 技术的开发者.企业决策人.云原生领域创业者为主,活动形式为演讲.动手实操 ...
- nginx导致获取客户端访问ip都是nginx服务器的地址问题解决
java 获取用户ip的方法 /** * 获得客户端 ip * @param request * @return */ public String getRemortIP(HttpServletReq ...
- mybatis-plus Date类型的参数 只有年月日 没有时分秒的解决办法
问题: 使用mybatis-plus 设计实体的时候 使用 Date inTime; 数据库里的时间2021-11-05 22:00:13 但java里的时间变成了2021-11-05 00: ...
- P1077-DP【黄】
昨天好几道题没做出来很郁闷,结果今天上来半小时不到就直接做出一道黄DP题了,不错,又有写题的冲动了. 这道题我一直被那个"因为方案数可能很多,请输出方案数对 1000007取模的结果.&qu ...
- /etc/profile,/etc/bashrc,~/.profile,~/.bashrc 的区别及使用
转载请注明出处: /etc/profile 为系统的全局环境变量设置,此文件为系统的每个用户设置环境信息 /etc/bashrc 为每一个运行bash shell的用户执行此文件.当bash ...
- 如何看待《李跳跳》APP因被腾讯公司发律师函称“不正当竞争”而无限期停止更新?
一波未平一波又起,继李跳跳无限期停更后,又一安卓神奇工具被下发律师函!近期各路安卓工具APP,被某讯大厂可谓是尽数剿灭~ 不难看出此次行动是"蓄谋已久"了.与李跳跳.大圣净化类似的 ...
- [转帖]java获取到heapdump文件后,如何快速分析?
https://www.jianshu.com/p/aaf56385766d 简介 在之前的OOM问题复盘之后,本周,又一Java服务出现了内存问题,这次问题不严重,只会触发堆内存占用高报警,没有 ...