2.14 haas506 2.0开发教程-高级组件库-modbus

2023-10-22 03:20

本文主要是介绍2.14 haas506 2.0开发教程-高级组件库-modbus,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

haas506 2.0开发教程-高级组件库-modbus

  • 1.Modbus slave仿真软件的使用
  • 2.案例
    • 案例一 对保持寄存器(holdRegister)进行数据读取和数据写入
    • 案例二 修改多个寄存器的值
    • 案例三 读取多个输入寄存器(InputRegisters)的值
  • 3.class-modbus

1.Modbus slave仿真软件的使用

需要先下载mbslave这个软件
链接: lmbslave官网.

  • Modbus是一种串行通信协议,是Modicon公司(现在的施耐德电气 Schneider Electric)于1979年为使用可编程逻辑控制器(PLC)通信而发表。Modbus已经成为工业领域通信协议的业界标准(Defacto),并且现在是工业电子设备之间常用的连接方式

  • modbus报文模型在这里插入图片描述

  • Modbus Slave是Modbus子设备模拟工具,模拟设备发送串口信号。可以仿真32个从设备/地址域。每个接口都提供了对excel报表的OLE自动化支持。帮助Modbus通讯设备开发人员进行Modbus通讯协议的模拟和测试,用于模拟、测试、调试Modbus通讯设备。与Modbus Poll的用户界面相同,支持功能01, 02, 03, 04, 05, 06, 15, 16, 22 和23,监视串口数据。

(1)打开仿真软件
在这里插入图片描述
(2)点击setup进行slave definition
在这里插入图片描述
 在slave definition中设置Slave ID(modbus从机ID)、Function(功能)、Address(地址)、Quantity(寄存器个数)、数据显示的格式等。
在这里插入图片描述
 (3)编辑寄存器的值
框选所有数据,选择Display - Unsigned在这里插入图片描述
  双击当前值,跳出一个Edit Register编辑框,修改寄存器的值在这里插入图片描述
全部修改后如下图所示:
    当前数据显示是无符号整型格式,若想修改数据的显示格式,可以点击软件窗口栏中的”Display"选择一个合适的数据格式。
在这里插入图片描述
例如选择“HEX"之后,数据显示如下图所示:
在这里插入图片描述
(4)Connection连接配置
  点击软件窗口栏中的Connection选项,进行连接配置。
在这里插入图片描述
 配置项:Connection选择”Serial Port“,Port需要选择设备的RS485的串口端口号,例如当前设备的RS485的端口号为COM26(从我的电脑-设备管理器获取),那么Port就选择Port26。波特率选择9600(Modbus采用串口通信时的串口波特率,默认为 9600)、数据位选择"8 Data bits"、奇偶校验选择”None Parity-不校验"、停止位选择"1 stop bit"。
在这里插入图片描述
 详细的配置项如下图所示:

在这里插入图片描述
设置完毕
在这里插入图片描述

2.案例

案例说明:

  • 使用Modbus slave工具,进行读写串口数据功能测试
  • 需要连接TTL与RS485串口,modbus使用RS485进行通信。

案例一 对保持寄存器(holdRegister)进行数据读取和数据写入

烧录代码

● main.py

import modbus
import utime as time
time.sleep(5)
#初始化modbus:RS485串口id为2,波特率为9600,奇偶校验为0,响应超时为1000ms
modbus.init(2,9600,0,1000)
#建立20个字节的数组
readBuf=bytearray(20)
#读保持寄存器:设备ID是1,寄存器起始地址为0,连续读入10个寄存器数据, 响应超时为1000ms
modbus.readHoldingRegisters(1,0,10,readBuf,1000)
print('read data from register:',readBuf)
#写保持寄存器:设备ID是1,寄存器地址为0,往地址为0的寄存器写入的数据为200, 响应超时1000ms
time.sleep(5)
modbus.writeHoldingRegister(1,0,200,1000)
modbus.readHoldingRegisters(1,0,10,readBuf,1000)
print("read register again:",readBuf)
#关闭modbus
modbus.deinit()

● board.json


{"name": "haas506","version": "1.0.0","io": {            "serial1":{"type":"UART","port":0,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial2":{"type":"UART","port":1,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial3":{"type":"UART","port":2,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"}},"debugLevel": "ERROR","repl":"enable","replPort":0}

● 日志输出
打开串口助手,查看打印的日志信息

read data from register:bytearray(b'\x00\x00\xff\xff\x01\x00\xff\xff\x02\x00\xff\xff\x03\x00\xff\xff\x04\x00\xff\xff')
read register again: bytearray(b'\xc8\x00\xff\xff\x01\x00\xff\xff\x02\x00\xff\xff\x03\x00\xff\xff\x04\x00\xff\xff')

注意:每个寄存器包含高低2个字节,寄存器数据按照大端字节序排列,大端就是一个数据的低字节在高地址,高字节在低地址。例如地址为0的寄存器的数据是0x00C8,那么输出的时候,先输出C8然后输出00。
在这里插入图片描述

案例二 修改多个寄存器的值

● main.py

  import modbusimport utimeutime.sleep(5)#修改前五个寄存器值,每个寄存器包含高低2个字节  #注意:(寄存器数据按照大端字节序排列) 大端就是一个数据的低字节在高地址,高字节在低地址modbus.init(2,9600,0,1000)readBuf=bytearray(10)#读保持寄存器:设备ID是1,寄存器起始地址为0,连续读入5个寄存器数据, 响应超时为1000msmodbus.readHoldingRegisters(1,0,5,readBuf,1000)print('read data from register:',readBuf)print("write data into register...")writeBuf=bytearray([0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00])#写多个保持寄存器:设备ID是1,寄存器的起始地址为0,待写入寄存器的数据为200, 响应超时1000msret=modbus.writeMultipleHoldingRegisters(1, 0, 5, writeBuf, 1000)print("----ret=",ret)# ----ret= (0, 0, 5, 0)#读保持寄存器:设备ID是1,寄存器起始地址为0,连续读入5个寄存器数据, 响应超时为1000msmodbus.readHoldingRegisters(1,0,5,readBuf,1000)print("read register again:",readBuf)modbus.deinit()

● board.json

{"name": "haas506","version": "1.0.0","io": {            "serial1":{"type":"UART","port":0,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial2":{"type":"UART","port":1,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial3":{"type":"UART","port":2,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"}},"debugLevel": "ERROR","repl":"enable","replPort":0}

● 日志

read data from registerbytearray(b'\x00\x00\xff\xff\x01\x00\xff\xff\x02\x00')
write data into register...
read register again: bytearray(b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00')

● 原始数据表
在这里插入图片描述

● 修改后的数据表
在这里插入图片描述

案例三 读取多个输入寄存器(InputRegisters)的值

在仿真软件中如下设置:
(1)创建一张表
在这里插入图片描述
(2)在slave definition中,选择Function中的 input Registers,注意当前设备的id是1。
在这里插入图片描述
(3)适当修改几个寄存器的值
在这里插入图片描述
● main.py


import modbus as mb
import utime as time
def modbus_test():#初始化串口2-->RS485,波特率9600,奇偶校验为0,超时1000ms mb.init(2, 9600, 0, 1000)readBuf = bytearray(10)#params:设备Id,寄存器起始地址, 所要读取的寄存器个数, readBuf, 超时时间设置ret = mb.readInputRegisters(1, 0, 5, readBuf, 200)print(ret)print('readBuf:',readBuf)mb.deinit()if __name__ == '__main__':while True:time.sleep(1)modbus_test()

● board.json

{"name": "haas506","version": "1.0.0","io": {            "serial1":{"type":"UART","port":0,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial2":{"type":"UART","port":1,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"},"serial3":{"type":"UART","port":2,"dataWidth":8,"baudRate":115200,"stopBits":1,"flowControl":"disable","parity":"none"}},"debugLevel": "ERROR","repl":"enable","replPort":0}

● 日志
代码执行过程中会输出两个值,分别是readInputRegisters()函数的状态返回值和前5个寄存器中的值

(0, 10)
readBuf: bytearray(b'\x00\x00\x01\x00\x02\x00\x03\x00\x04\x00')
(0, 10)
readBuf: bytearray(b'\x00\x00\x01\x00\x02\x00\x03\x00\x04\x00')
(0, 10)
......

3.class-modbus

initdeinitwriteHoldingRegisterwriteMultipleHoldingRegisterswriteCoilwriteMultipleCoilsreadHoldingRegistersreadInputRegistersreadDiscreteInputsreadCoilsrecvsend
初始化Modbus关闭Modbus写单个保持寄存器写多个保持寄存器写单个线圈写多个线圈读多个保持寄存器值读取多个输入寄存器值读取多个离散输入值读取多个线圈值从机读取主机数据从机发行响应数据到主机

modbus.init(nodeName) - 初始化Modbus

函数功能:根据board.json中设备节点的配置初始化Modbus总线
参数说明:
参数类型必选参数?说明
nodeNamestringboard.json配置文件中Modbus设备节点的名称
board.json中的UART类型设备属性配置项说明如下:
属性字段数据类型属性值必选项?字段说明
typestringMODBUS节点类型,表明当前节点为Modbus节点,固定设置为MODBUS
modeint0、1Modbus物理通道类型,0表示串口,1表示TCP以太网。目前仅支持串口通道
portint0、1 等Modbus采用串口通信时的串口号
baudrateint9600、115200等Modbus采用串口通信时的串口波特率,默认为 9600
timeoutint200, 1000等Modbus采用串口通信时的超时参数,单位是毫秒(ms),默认为200ms
parityint0、1、2奇偶校验设置,默认 0
返回值:打开Modbus设备成功返回0;打开UART设备失败返回失败错误码

modbus.deinit() - 关闭Modbus

函数功能:反初始化Modbus总线
注意事项:需确保要关闭的Modbus处于init状态
参数说明:无
返回值:反初始化Modbus设备成功返回0;反初始化Modbus设备失败返回失败错误码

modbus.writeHoldingRegister(slave_addr, register_addr, register_value, timeout) - 写单个保持寄存器

函数功能:向从机写单个保持寄存器
注意事项:需确保此UART处于open状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
register_addrint写寄存器的地址
register_valueint写寄存器的数据
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的4元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
resp_addr响应地址
resp_value响应数据
exception_code响应异常代码

modbus.writeMultipleHoldingRegisters(slave_addr, start_addr, reg_quantity, data, timeout) - 写多个保持寄存器

函数功能:向从机多个保持寄存器中写入数据
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
register_addrint待写寄存器的起始地址
reg_quantityint待写寄存器的数量,表示操作多少个寄存器
databytearray写寄存器的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的4元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
resp_addr响应地址
resp_quantity真实完成寄存器操作的数量
exception_code响应异常代码

modbus.writeCoil(slave_addr, coil_addr, coil_value, timeout) - 写单个线圈

函数功能:向从机某个线圈中写入数据
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
coil_addrint写线圈的地址
coil_valueint写线圈的数据
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的4元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
resp_addr响应地址
resp_value响应数据
exception_code响应异常代码

modbus.writeMultipleCoils(slave_addr, start_addr, reg_quantity, data, timeout) - 写多个线圈

函数功能:向从机多个线圈中写入数据
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
start_addrint写线圈的地址
reg_quantityint待写寄存器的数量,表示操作多少个寄存器
databytearray写寄存器的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的4元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
resp_addr响应地址
resp_quantity真实完成寄存器操作的数量
exception_code响应异常代码
modbus.readHoldingRegisters(slave_addr, start_addr, reg_quantity, data, timeout) - 读多个保持寄存器值
函数功能:向从机读取多个保持寄存器的值
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
start_addrint待读寄存器的起始地址
reg_quantityint待读寄存器的数量,表示操作多少个寄存器
databytearray读寄存器获得的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
respond_count读取到数据的字节数

modbus.readInputRegisters(slave_addr, start_addr, reg_quantity, data, timeout) - 读取多个输入寄存器值

函数功能:向从机读取多个输入寄存器的值
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
start_addrint待读寄存器的起始地址
reg_quantityint待读寄存器的数量,表示操作多少个寄存器
databytearray读寄存器获得的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
respond_count读取到数据的字节数
modbus.readDiscreteInputs(slave_addr, start_addr, reg_quantity, data, timeout) - 读取多个离散输入值
函数功能:向从机读取多个输入寄存器的值
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
start_addrint待读寄存器的起始地址
reg_quantityint待读寄存器的数量,表示操作多少个寄存器
databytearray读寄存器获得的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
respond_count读取到数据的字节数
modbus.readCoils(slave_addr, start_addr, reg_quantity, data, timeout) - 读取多个线圈值
函数功能:向从机读取多个线圈的值
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint请求的从机地址,0代表广播
start_addrint待读寄存器的起始地址
reg_quantityint待读寄存器的数量,表示操作多少个寄存器
databytearray读寄存器获得的数据,每个寄存器包含高低两个字节,高位在前,低位在后
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
respond_count读取到数据的字节数

modbus.recv() - 从机读取主机数据

函数功能:该函数仅供从机使用,用来接收主机发送的请求命令。接收的超时参数由 modbus.init 初始化时指定的 timeout 决定,默认为200毫秒。
注意事项:需确保此Modbus处于init状态
参数说明:
参数类型必选参数?说明
slave_addrint从机地址
databytearray响应数据,格式为: 响应代码 + 响应数据
timeoutint请求超时时间,单位是毫秒(ms)
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
length发送的响应帧的长度,包含CRC校验长度

modbus.send(slave_addr, data, timeout) - 从机发行响应数据到主机

函数功能:该函数仅供从机使用,用于发送响应数据给主机。接收的超时参数由 modbus.init 初始化时指定的 timeout 决定,默认为200毫秒。
注意事项:需确保此Modbus处于init状态
函数原型:modbus.send()
参数说明: 空
返回值:返回一个tuple类型的2元组,元组中的条目格式为:
status请求状态,0表示成功,其他表示失败,具体含义参考 STATUS
bytes接收到的数据,包含CRC校验数据

这篇关于2.14 haas506 2.0开发教程-高级组件库-modbus的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/258769

相关文章

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

React+TS前台项目实战(十七)-- 全局常用组件Dropdown封装

文章目录 前言Dropdown组件1. 功能分析2. 代码+详细注释3. 使用方式4. 效果展示 总结 前言 今天这篇主要讲全局Dropdown组件封装,可根据UI设计师要求自定义修改。 Dropdown组件 1. 功能分析 (1)通过position属性,可以控制下拉选项的位置 (2)通过传入width属性, 可以自定义下拉选项的宽度 (3)通过传入classN

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

Python应用开发——30天学习Streamlit Python包进行APP的构建(9)

st.area_chart 显示区域图。 这是围绕 st.altair_chart 的语法糖。主要区别在于该命令使用数据自身的列和指数来计算图表的 Altair 规格。因此,在许多 "只需绘制此图 "的情况下,该命令更易于使用,但可定制性较差。 如果 st.area_chart 无法正确猜测数据规格,请尝试使用 st.altair_chart 指定所需的图表。 Function signa

Steam邮件推送内容有哪些?配置教程详解!

Steam邮件推送功能是否安全?如何个性化邮件推送内容? Steam作为全球最大的数字游戏分发平台之一,不仅提供了海量的游戏资源,还通过邮件推送为用户提供最新的游戏信息、促销活动和个性化推荐。AokSend将详细介绍Steam邮件推送的主要内容。 Steam邮件推送:促销优惠 每当平台举办大型促销活动,如夏季促销、冬季促销、黑色星期五等,用户都会收到邮件通知。这些邮件详细列出了打折游戏、

X-AnyLabeling使用教程

1.AI 模型自动分割标注使用教程 2.AI 模型自动目标检测标注使用教程

WDF驱动开发-WDF总线枚举(一)

支持在总线驱动程序中进行 PnP 和电源管理 某些设备永久插入系统,而其他设备可以在系统运行时插入和拔出电源。 总线驱动 必须识别并报告连接到其总线的设备,并且他们必须发现并报告系统中设备的到达和离开情况。 总线驱动程序标识和报告的设备称为总线的 子设备。 标识和报告子设备的过程称为 总线枚举。 在总线枚举期间,总线驱动程序会为其子 设备创建设备对象 。  总线驱动程序本质上是同时处理总线枚

青龙面板2.9之Cdle傻妞机器人编译教程

看到有的朋友对傻妞机器人感兴趣,这里写一下傻妞机器人的编译教程。 第一步,这里以linux amd64为例,去官网下载安装go语言安装包: 第二步,输入下方指令 cd /usr/local && wget https://golang.google.cn/dl/go1.16.7.linux-amd64.tar.gz -O go1.16.7.linux-amd64.tar.gz

青龙面板部署通用教程,含服务器、路由器、X86等部署方法

1. 拉取镜像/更新镜像 docker pull whyour/qinglong:latest 2. 删除镜像 docker rmi whyour/qinglong:latest 3. 启动容器 普通服务器 docker run -dit \-v $PWD/ql/config:/ql/config \-v $PWD/ql/log:/ql/log \-v $PWD/ql/db: