当前位置: 首页 > article >正文

吃透modbus协议

转载自b站

这节课带你吃透Modbus通信协议_哔哩哔哩_bilibili

理论

Modbus是一种应用层的报文传输协议,Modbus协议不是唯一的,有三种形式:RTU、ASCLL、TCP,他们之间既有共同点,又有不同点。


一、Modbus RTU通信协议为例:

01.存储区(数据存储区、文件存储区)

有四种:
输出线圈: 代号是0
输入线圈: 代号是1
输出(写)寄存器: 代号是4
输入(读)寄存器: 代号是3
线圈:最小单位bool,要么存的是1,要么是0。
寄存器:16位,2个字节
总结:如果用bool,就用线圈;如果要存数据,就用寄存器。


02.存储区范围

5位和6位 标准地址(5位)和扩展地址
标准地址(5位)
YXXXX
Y表示哪个存储区,取值范围0、1、3、4
X取值范围0-9
输出线圈: 代号是0,存储范围:00001-09999
输入线圈: 代号是1,存储范围:10001-19999
输出(写)寄存器: 代号是4,存储范围:40001-49999
输入(读)寄存器: 代号是3,存储范围:30001-39999
扩展地址(6位)
YXXXXX
输出线圈:存储范围:000001-065536
输入线圈: 代号是1,存储范围:100001-165536
输出(写)寄存器: 代号是4,存储范围:400001-465536
输入(读)寄存器: 代号是3,存储范围:300001-365536


03.通讯协议

八种动作:四个读、四个写
读输出线圈:功能码:01(十六进制)
读输入线圈:功能码:02
读输出寄存器:功能码:03
读输入寄存器:功能码:04
写单个输出线圈:功能码:05
写单个输出寄存器:功能码:06
写多个输出线圈:功能码:0f
写多个输出寄存器:功能码:10

报文格式:
从站地址(设备编号1byte)+功能码(1byte)+数据(N个byte)+校验(2个byte)

对于读取来说:
从站地址(设备编号):为了区分是哪个设备。
功能码:是为了确定干什么。
数据:具体干嘛的细节。
校验:验证

对于写入来说
从站地址(设备编号):为了区分是哪个设备。
功能码:是为了确定干什么。
数据:具体干嘛的细节(更多,多了个写入的具体数值)。
校验:验证

以下是一个例子:
监听com24端口:采集湿度和温度
上位机发送:01 03 00 00 00 02 C4 0B
01 站地址
03 功能码:读输出寄存器
00 00 起始寄存器
00 02 寄存器长度
C4 0B CRC校验
设备返回:01 03 04 01 46 01 3B 5A 59
01 站地址
03 功能码:读输出寄存器
04 字节计数(字节总数)
01 46 01 3B 具体数据4个字节
5A 59 CRC校验

其中
01 46 表示湿度,十六进制,对应的十进制为326,326/10=32.6
01 3B表示温度,十六进制,对应的十进制为315,315/10=31.5
ModbusCRC校验码由算法算出,不用深究怎么算,能算出来就行

温度的Modbus地址是多少?40001->绝对地址
湿度的Modbus地址是多少?40002->绝对地址
协议上不会使用绝对地址,因为03已经说明了是读输出寄存器。
通讯协议当中用的全是相对地址,相对地址是从0开始的。

二、常用功能码

Modbus中常用的功能码有8个,可以分为位操作和字操作两类,如下表所示:
在这里插入图片描述

03. 读线圈寄存器(01H)

功能码01H读取Modbus从机中线圈寄存器的状态,可以是单个寄存器,或者多个连续的寄存器。

发送

假设从机地址为01H,读取的线圈寄存器的起始地址为0017H,读取38个寄存器,指令如下所示:

读线圈寄存器指令
在这里插入图片描述


响应

各线圈的状态与数据内容的每个bit对应,1代表ON,0代表OFF。如果查询的线圈数量不是8的倍数,则在最后一个字节的高位补0。

读线圈状态的返回结果
在这里插入图片描述

 解读:其中,第一个字节CDH对应线圈0017H到001E的状态,转为二进制是11001101,其中bit0对应0017H,bit7对应001E,如下表所示:

线圈0017H到001EH的状态
在这里插入图片描述

最后一个字节为1BH,对应线圈0037H到003CH的状态,转为二进制是00011011,其中bit0对应0037H,bit5对应003CH,其余两位用0填充:

线圈0037H到003CH的状态
在这里插入图片描述

读线圈寄存器——01H

从站发送读取格式:

从站地址功能码起始地址高位起始地址低位读取数量高位读取数量低位CRC高位CRC低位
0x010x010x000x000x000x02BDCB

主站返回读取格式:

从站地址功能码返回字节数data1CRC高位CRC低位
0x010x010x010x005188

解读:
发送格式:
从站地址:01
功能码:01
起始地址: 00 00
读取数量: 00 02
CRC校验:BD CB
接收格式:
从站地址:01
功能码:01
返回字节数: 01
Data数据: 00(一个byte有8个bit,读取两个bit返回会补成字节用一个byte返回读取两个bit就行)
CRC校验:BD CB
 

例如:01 01 01 00 51 88 

            01 01 01 01 90 48

04. 读离散输入寄存器(02H)

功能码02H读取Modbus从机中离散输入寄存器的状态,可以是单个寄存器,或者多个连续的寄存器。

发送

假设从机地址为01H,读取的离散输入寄存器的起始地址为00C4H,读取22个寄存器,指令如下所示:

读离散输入寄存器指令:
在这里插入图片描述

响应

各个离散输入寄存器的状态与数据内容的每个bit对应,1代表ON,0代表OFF。如果查询的线圈数量不是8的倍数,则在最后一个字节的高位补0。

读离散输入寄存器的返回结果
在这里插入图片描述

其中,第一个字节ACH对应00C4H到00CBH寄存器的状态,转为二进制是10101100,其中bit0对应00C4H,bit7对应00CB,如下所示:

寄存器00C4H到00CBH的状态
在这里插入图片描述

最后一个字节为35H,对应寄存器00D4H到00D9H的状态,转为二进制是00110101,其中bit0对应00D4H,bit5对应00D9H,其余两位用0填充,

寄存器00D4H到00D9H的状态在这里插入图片描述

05. 读保持寄存器(03H)

功能码03H读取Modbus从机中保持寄存器的数据,可以是单个寄存器,或者多个连续的寄存器。

发送

假设从机地址为01H,读取的保持寄存器的起始地址为006BH,读取3个寄存器,指令如表5.1所示:

读保持寄存器指令:
在这里插入图片描述

响应

每个保持寄存器的长度为2个字节。保持寄存器之间,低地址寄存器先传输,高地址寄存器后传输。单个保持寄存器,高字节数据先传输,低字节数据后传输。

读保持寄存器的返回结果
在这里插入图片描述

06. 读输入寄存器(04H)

功能码04H读取Modbus从机中输入寄存器的数据,可以是单个寄存器,或者多个连续的寄存器。

发送

假设从机地址为01H,读取的保持寄存器的起始地址为006BH,读取2个寄存器,指令如下所示:

读输入寄存器指令:
在这里插入图片描述

响应

每个输入寄存器的长度为2个字节。输入寄存器之间,低地址寄存器先传输,高地址寄存器后传输。单个输入寄存器,高字节数据先传输,低字节数据后传输。

读输入寄存器的返回结果
在这里插入图片描述

07. 写单个线圈寄存器(05H)

功能码05H写单个线圈寄存器,FF00H请求线圈处于ON状态,0000H请求线圈处于OFF状态。

发送

假设从机地址为01H,线圈寄存器的地址为00ACH,使其处于ON状态的指令如下所示:

写单个线圈指令:
在这里插入图片描述

响应

如果写入成功,返回发送的指令,即010500ACFF004C1B。

写入:01 05 00 00 FF 00 8C 3A  

返回:01 05 00 00 FF 00 8C 3A 

08. 写单个保持寄存器(06H)

功能码06H写单个保持寄存器。

发送

假设从机地址为01H,保持寄存器的地址为0001H,数据位0003H,指令如下所示:

写单个保持寄存器指令:
在这里插入图片描述

响应

如果写入成功,返回发送的指令,即010600010003980B。

09. 写多个线圈寄存器(0FH)
功能码0FH写多个线圈寄存器。如果对应的数据位为1,表示线圈状态为ON;如果对应的数据位为0,表示线圈状态为OFF。线圈寄存器之间,低地址寄存器先传输,高地址寄存器后传输。单个线圈寄存器,高字节数据先传输,低字节数据后传输。如果写入的线圈寄存器的个数不是8的倍数,则在最后一个字节的高位补0。

发送

假设从机地址为01H,线圈寄存器的起始地址为0013H,写入10个寄存器,指令如下所示:

写入多个线圈寄存器指令

在这里插入图片描述

其中,CDH对应线圈0013H到001AH的内容,01H对应线圈001B到001CH的内容,未使用位用0填充。

此时,线圈寄存器的内容如下所示:

线圈寄存器0013H到001CH的内容
在这里插入图片描述

响应

如果写入成功,返回写入的寄存器数量

写多个线圈寄存器的返回结果
在这里插入图片描述

10. 写多个保持寄存器(10H)

功能码10H写多个保持寄存器,其中每个保持寄存器的长度为两个字节。

发送

假设从机地址为01H,保持寄存器的起始地址为0001H,写入2个寄存器,指令如下所示:

写入多个保持寄存器指令
在这里插入图片描述

返回数据格式不返回Data数据

详情:写入数量为2的线圈,其中data1,data2,data3,data4数据两个代表一个单精度数据
0C 02表示一个3074 ,12 45代表一个4677

响应

如果写入成功,返回写入的寄存器数量,

写多个保持寄存器的返回结果
在这里插入图片描述

写多个线圈——0FH
从站地址功能码起始地址高位起始地址低位写入数量高位写入数据低位字节长度data1CRC高位CRC低位
0x010x0F0x000x000x000x020x010x039E96

主站返回读取格式:
响应:

从站地址功能码起始地址高位起始地址低位写入数量高位写入数据低位CRC高位CRC低位
0x010x0F0x000x000x000x029E96

返回数据格式不返回Data数据

详情:写入数量为2的线圈,其中data1数据0x03代表0000 0011 将连续两个线圈置为ON

http://www.lryc.cn/news/2412651.html

相关文章:

  • mysql数据库安装(详细)
  • 关于STM32
  • 字母数字-ASCII码表(最详细,最直观)
  • 《Scientific Reports》2024最新投稿经验
  • 一个很好用的BUG收集工具------Bugly
  • Kettle调优教程(推荐收藏)
  • 初学Vue(全家桶)-第16天(vuex):vuex简介
  • 全面解析ASCII码:ASCII码表、大小顺序与实际应用详解
  • OpenSSL命令行快速入门
  • XML基础入门:关于XML解析
  • ELK介绍及架构分析
  • Vue.js学习
  • Zookeeper详解(最详细的zookeeper解析+项目实例)
  • lodash的用法详解
  • Base64解码
  • 推荐一款效率类小工具--utools
  • RocketMQ高级原理
  • C语言Switch语句的case用法详解
  • Zabbix最详细教程Ubuntu部署Zabbix6.0[图文]
  • “多重人格”的操作系统——openEuler
  • RabbitMQ详解:消息队列的原理、应用与最佳实践
  • 了解伽马(GAMMA、伽马值、光度、灰度系数)
  • 【计算机视觉 | 图像分割】arxiv 计算机视觉关于图像分割的学术速递(12 月 1 日论文合集)(上)
  • UniApp入门指南以及组件的使用
  • 从零基础学Go(六)——Go的复杂数据结构(下)
  • python编程:从入门到精通,python编程教学入门教程
  • OpenWrt网络配置详解
  • 【OpenCV】简介
  • 医学图像中的窗宽(Window Width,WW)和窗位(Window Level,WL)
  • Stream 流常见基本操作