python - Python Struct.Unpack 中的对齐/打包

标签 python struct unpack

我有一个硬件以固定长度发送数据:2字节,1字节,4字节,4字节,2字节,4字节,总共17字节。如果我将格式更改为 18 字节,代码可以工作,但值不正确。

format = '<2s1s4s4s2s4s'
print(struct.calcsize(format))
print(len(hardware_data))
splitdata = struct.unpack(format,hardware_data)

输出是 17、18,并且由于不匹配而出现错误。我认为这是由对齐引起的,但我不确定,并且我尝试过的任何方法都无法解决此问题。下面是几个典型的字符串,如果我 print(hardware_data) 我注意到“R”和“n”字符,但我不确定如何处理。

b'\x18\x06\x00R\x1f\x01\x00\x00\x00\x00\x00\xd8\xff\x00\x00\x00\x00\x80'

b'\x18\x06\x00R\x1f\x01\x00\x00\x00\x00\x00\n\x00\x00\x00\x00\x00\x80'

最佳答案

很可能发送数据的任何内容都以您不期望的方式填充它。

例如,如果前四个字节字段应该表示 int,则 C 结构体填充规则将需要一个填充字节,位于一个字节字段之后(以将下一个四字节字段与四字节对齐)。因此,只需显式添加填充字节,将格式字符串更改为:

format = '<2s1sx4s4s2s4s'

其中的 x 表示“我期望这里有一个字节,但它是填充,不要将其解压为任何内容。”填充字节可能属于其他地方(我不知道你的硬件在做什么);我注意到两个示例中的第三个字节都是 NUL (\0) 字节,但我认为填充的位置是 'R',所以您可能想要:

format = '<2sx1s4s4s2s4s'

相反。或者它可能在其他地方(不知道哪些字段是硬件结构中的 char 数组,哪些是具有对齐要求的较大类型,这是不可能说的)。重点是,您的硬件正在发送 18 个字节;找出哪个是垃圾,然后将 x 填充字节放在适当的位置。

旁注:bytes 对象的 repr 将使用 ASCII 或更简单的 ASCII 转义(如果可用)。这就是为什么您会在输出中看到 R\nb'R'b'\x52' 是等效的文字,b'\n'b'\x0a 也是如此' 并且 Python 选择使用“更易读”的版本(当 bytes 实际上只是 ASCII 时,这更具可读性)。

关于python - Python Struct.Unpack 中的对齐/打包,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58332625/

相关文章:

python - 无法回滚带有 MySQL 非事务性更改表的 Django

python - 无法使用 bs4 从 BSE 网站上抓取特定信息

c - 使用 union 时将标签字段值分配给结构

perl - 如何解压 "pairwise reversed"命令BA DC中的字节?

class - 如何使用类声明将一流模块解压缩为对象模块?

python - 在 Python 中使用 pandas 对过滤后的数据应用过滤器

python - 使用额外的索引键展平 DataFrame 嵌套列表/数组(用于时间序列)

c - 具有必须在 C 中的 .h 文件中公开的属性的私有(private)结构

在位域整数与结构之间选择?

python - 为什么解包结构会产生元组?