首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >PyModbus无法通过TCP读取RTU中的输入寄存器

PyModbus无法通过TCP读取RTU中的输入寄存器
EN

Stack Overflow用户
提问于 2020-10-27 08:16:46
回答 1查看 542关注 0票数 0

我尝试使用pymodbus读取Modbus信号,但得到错误,来源如下:

当服务器响应TCP时,有一个MAC地址通知功能。怎样才能无错误地进行Modbus通信?如何删除TCP报头部分?

该手册的地址如下:https://www.eztcp.com/en/download/pds_files/an_ezmanager_en.pdf

代码语言:javascript
复制
from pymodbus.client.sync import ModbusTcpClient
from pymodbus.transaction import ModbusRtuFramer as ModbusFramer
import time

import logging
logging.basicConfig()
log = logging.getLogger()
log.setLevel(logging.DEBUG)

def run_read_data(ip, port_num, unit_num):
    while True:
        client = ModbusTcpClient(ip, port=port_num, framer=ModbusFramer)
        con = client.connect()
        while con :
            rr = client.read_input_registers(0,10, unit=unit_num)        
            try:
                print(rr,rr.registers)

            except Exception as e:
                print(f"{type(e).__name__}: {e}")
  
                time.sleep(2)
                client.close()
                time.sleep(2)
                break
            time.sleep(1)
        time.sleep(1)

if __name__ == "__main__":
    ip = "x.x.x.x" 
    port_num = x
    unit_num = 0x1
    run_read_data(ip, port_num,unit_num)

结果是这样的:

代码语言:javascript
复制
DEBUG:pymodbus.transaction:Current transaction state - IDLE
DEBUG:pymodbus.transaction:Running transaction 1
DEBUG:pymodbus.transaction:SEND: 0x18 0x4 0x0 0x0 0x0 0xa 0x72 0x4
DEBUG:pymodbus.client.sync:New Transaction state 'SENDING'
DEBUG:pymodbus.transaction:Changing transaction state from 'SENDING' to 'WAITING FOR REPLY'
DEBUG:pymodbus.transaction:Changing transaction state from 'WAITING FOR REPLY' to 'PROCESSING REPLY'
DEBUG:pymodbus.transaction:RECV: 0x30 0x30 0x33 0x30 0x66 0x39 0x31 0x31 0x34 0x30 0x38 0x35 0xd 0xa 0x18 0x4 0x14 0xe8 0xf9 0x2 0x45 0xf3 0x88 0x1 0x22
DEBUG:pymodbus.framer.rtu_framer:CRC invalid, discarding header!!
DEBUG:pymodbus.framer.rtu_framer:Resetting frame - Current Frame in buffer - 0x30 0x30 0x33 0x30 0x66 0x39 0x31 0x31 0x34 0x30 0x38 0x35 0xd 0xa 0x18 0x4 0x14 0xe8 0xf9 0x2 0x45 0xf3 0x88 0x1 0x22
DEBUG:pymodbus.framer.rtu_framer:Frame check failed, ignoring!!
DEBUG:pymodbus.framer.rtu_framer:Resetting frame - Current Frame in buffer -
DEBUG:pymodbus.transaction:Getting transaction 24
DEBUG:pymodbus.transaction:Changing transaction state from 'PROCESSING REPLY' to 'TRANSACTION_COMPLETE'
AttributeError: 'ModbusIOException' object has no attribute 'registers'
DEBUG:pymodbus.transaction:Current transaction state - IDLE
DEBUG:pymodbus.transaction:Running transaction 1
DEBUG:pymodbus.transaction:SEND: 0x18 0x4 0x0 0x0 0x0 0xa 0x72 0x4
DEBUG:pymodbus.client.sync:New Transaction state 'SENDING'
DEBUG:pymodbus.transaction:Changing transaction state from 'SENDING' to 'WAITING FOR REPLY'
DEBUG:pymodbus.transaction:Changing transaction state from 'WAITING FOR REPLY' to 'PROCESSING REPLY'
DEBUG:pymodbus.transaction:RECV: 0x30 0x30 0x33 0x30 0x66 0x39 0x31 0x31 0x34 0x30 0x38 0x35 0xd 0xa 0x18 0x4 0x14 0xe9 0x9f 0x2 0x45 0xf3 0xd5 0x1 0x22
DEBUG:pymodbus.framer.rtu_framer:CRC invalid, discarding header!!
DEBUG:pymodbus.framer.rtu_framer:Resetting frame - Current Frame in buffer - 0x30 0x30 0x33 0x30 0x66 0x39 0x31 0x31 0x34 0x30 0x38 0x35 0xd 0xa 0x18 0x4 0x14 0xe9 0x9f 0x2 0x45 0xf3 0xd5 0x1 0x22
DEBUG:pymodbus.framer.rtu_framer:Frame check failed, ignoring!!
DEBUG:pymodbus.framer.rtu_framer:Resetting frame - Current Frame in buffer -
DEBUG:pymodbus.transaction:Getting transaction 24
DEBUG:pymodbus.transaction:Changing transaction state from 'PROCESSING REPLY' to 'TRANSACTION_COMPLETE'
AttributeError: 'ModbusIOException' object has no attribute 'registers'
EN

回答 1

Stack Overflow用户

发布于 2020-10-27 09:13:11

您收到的消息是无效的,您可以看到CRC检查失败-问题出在modbusTCP服务器上。因为response没有属性registers,所以会导致异常。但是,如果您想要查看错误是什么样子,您应该打印rr,而不打印任何其他内容。

检查是否有错误的好方法是使用isError()方法:

代码语言:javascript
复制
if rr.isError():
    # handle error
else:
    # proceed with response

如果你正在尝试解码消息,你应该使用BinaryPayloadDecoder,如下所示:

代码语言:javascript
复制
decoder = BinaryPaloadDecoder.fromRegisters(rr.registers)
value = decoder.decode_32bit_float()

完整参考here

不幸的是,我不知道你说的“如何删除TCP报头部分”是什么意思。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/64546682

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档