![](/img/trans.png)
[英]Pymodbus TCP `read_holding_registers` returns stale/old data
[英]Pymodbus read holding registers
我被指派在沒有任何文件的情況下執行任務。 我在從 MODBUS 讀取數據時遇到問題。 這是我能夠創建的腳本:
from pymodbus.constants import Endian
from pymodbus.payload import BinaryPayloadDecoder
from pymodbus.payload import BinaryPayloadBuilder
from pymodbus.client.sync import ModbusTcpClient
client = ModbusTcpClient('X.X.X.X')
connection = client.connect()
request = client.read_holding_registers(12606,2)
result = request.registers
decoder = BinaryPayloadDecoder.fromRegisters(result, Endian.Big, wordorder=Endian.Little)
print "Counter1: %0.2f" % decoder.decode_32bit_float()
request = client.read_holding_registers(12482,2)
result = request.registers
decoder = BinaryPayloadDecoder.fromRegisters(result, Endian.Big, wordorder=Endian.Little)
print "Counter2: %0.2f" % decoder.decode_32bit_float()
client.close()
一切看起來都很好,但是計數器上的數據與腳本中的數據不同,例如:
Value on the counter : 39558853.30 (value is decimal)
Value from the script: 58853.30
(value is decimal)
Read input registers (HEX): E54D 4765
這就是地址文檔的樣子”
P 12458 Q2 4\DW12458 = 1\ND20_Q2\P(F)
Q 12462 Q2 4\DW12462 = 1\ND20_Q2\Q(F)
S 12466 Q2 4\DW12466 = 1\ND20_Q2\S(F)
I 12470 Q2 4\DW12470 = 1\ND20_Q2\I(F)
U 12474 Q2 4\DW12474 = 1\ND20_Q2\U(F)
f 12478 Q2 4\DW12478 = 1\ND20_Q2\f(F)
EP_POB 12482 Q2 4\DW12482 = 1\ND20_Q2\EP_POB(F)
EP_ODD 12486 Q2 4\DW12486 = 1\ND20_Q2\EP_ODD(F)
EQ_IND 12490 Q2 4\DW12490 = 1\ND20_Q2\EQ_IND(F)
EQ_POJ 12494 Q2 4\DW12494 = 1\ND20_Q2\EQ_POJ(F)
THDVL1 12498 Q2 4\DW12498 = 1\ND20_Q2\THDVL1(F)
THDVL2 12502 Q2 4\DW12502 = 1\ND20_Q2\THDVL2(F)
THDVL3 12506 Q2 4\DW12506 = 1\ND20_Q2\THDVL3(F)
THDIL1 12510 Q2 4\DW12510 = 1\ND20_Q2\THDIL1(F)
THDIL2 12514 Q2 4\DW12514 = 1\ND20_Q2\THDIL2(F)
THDIL3 12518 Q2 4\DW12518 = 1\ND20_Q2\THDIL3(F)
UL1 12522 Q2 4\DW12522 = 1\ND20_Q2\UL1(F)
UL2 12526 Q2 4\DW12526 = 1\ND20_Q2\UL2(F)
UL3 12530 Q2 4\DW12530 = 1\ND20_Q2\UL3(F)
IL1 12534 Q2 4\DW12534 = 1\ND20_Q2\IL1(F)
IL2 12538 Q2 4\DW12538 = 1\ND20_Q2\IL2(F)
IL3 12542 Q2 4\DW12542 = 1\ND20_Q2\IL3(F)
PL1 12546 Q2 4\DW12546 = 1\ND20_Q2\PL1(F)
PL2 12550 Q2 4\DW12550 = 1\ND20_Q2\PL2(F)
PL3 12554 Q2 4\DW12554 = 1\ND20_Q2\PL3(F)
QL1 12558 Q2 4\DW12558 = 1\ND20_Q2\QL1(F)
QL2 12562 Q2 4\DW12562 = 1\ND20_Q2\QL2(F)
QL3 12566 Q2 4\DW12566 = 1\ND20_Q2\QL3(F)
S1 12570 Q2 4\DW12570 = 1\ND20_Q2\S1(F)
S2 12574 Q2 4\DW12574 = 1\ND20_Q2\S2(F)
S3 12578 Q2 4\DW12578 = 1\ND20_Q2\S3(F)
我改進了你的代碼如下:
from pymodbus.constants import Endian
from pymodbus.payload import BinaryPayloadDecoder
from pymodbus.client.sync import ModbusTcpClient
def validator(instance):
if not instance.isError():
'''.isError() implemented in pymodbus 1.4.0 and above.'''
decoder = BinaryPayloadDecoder.fromRegisters(
instance.registers,
byteorder=Endian.Big, wordorder=Endian.Little
)
return float('{0:.2f}'.format(decoder.decode_32bit_float()))
else:
# Error handling.
print("There isn't the registers, Try again.")
return None
client = ModbusTcpClient('X.X.X.X', port=502) # Specify the port.
connection = client.connect()
if connection:
request = client.read_holding_registers(12606, 2, unit=1) # Specify the unit.
data = validator(request)
print(data)
request = client.read_holding_registers(12482, 2, unit=1) # Specify the unit.
data = validator(request)
print(data)
client.close()
else:
print('Connection lost, Try again')
[注意]:
您是否確保所需的 float32 解碼?
byteorder=Endian.Big, wordorder=Endian.Big
byteorder=Endian.Big, wordorder=Endian.Little
byteorder=Endian.Little, wordorder=Endian.Big
byteorder=Endian.Little, wordorder=Endian.Little
設置unit_ID :
unit
是1
作為默認 ID。[更新]:
也許您需要讀取和解碼為12482
寄存器地址處的double/float64值,因為我認為當文檔中所需的寄存器是12482
並且下一個寄存器是12846
,因此我們需要讀取4regs ( float64/double ):
request = client.read_holding_registers(12482, 4, unit=1)
和
return float('{0:.2f}'.format(decoder.decode_64bit_float()))
我很確定 39558853.30 的值太大了,無法存儲在 IEEE 單精度浮點數中。 有 7.22 位精度,該數字需要 9 位。 我做了一些在 C# 中將值分配給 float 和 double 值的實驗證實了這一點。
這讓我相信:
1) 像 Benyamin Jafari 建議的那樣,您需要讀取四個寄存器。 但是,十六進制中的該數字(作為雙精度數)是 0x4182dcf62a666666,它似乎與您正在閱讀的任何數據都不對應。
或者
2) 它也可能作為 UINT32 返回,必須按 (1/100) 縮放才能為您提供計數器顯示的內容。 0xE54D4765 = 3847047013 => 按 1/100.0 = 38470470.13 縮放,這與您在櫃台上看到的很接近。 根據我的經驗,這是 Modbus 中的常見做法。
或者
3)他們使用其他(非標准)格式來表示數據。
你能給我們產品名稱,型號等嗎?
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.