python结构体字节对齐_Python及字节对齐的问题

该结构大致如下:

struct TestStruct

{

int data1;

char data2;

char data3;

};

对应的Python代码:

import struct

s = struct.unpack("icc", buf) #buf是从网络接收的字节流

结果却报“Unpack str size does not match format”错。

很明显是C++ struct产生的size和Python解码所需的不同。于是检查C++的struct size:

printf ("size=%d\n", sizeof(TestStruct));

结果得8。

struct.calcsize("icc")

结果却是6。

仔细想想,icc这种排列方式在字段间的确不会产生padding字节,也就是说,python的结果是对的。但为什么C++的结果会是8呢?原来C++的字节对齐,除了struct内部需要字节对齐之外,struct变量本身也是需要字节对齐的,这是为了当生成一个struct数组的时候仍然能够保证所有字段的字节对齐。因此,像icc这种本身没有padding,但整体需要padding的情形,编译器会在整个struct的末尾加上padding字节(在这里是2个字节),也正是这2个字节导致了Python的解码错误。

幸亏Python已经考虑到了这个问题,在Python struct module document的最后一段的Hint

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
根据引用[1]中提到的ctypesgen生成的结构体,ctypes创建的结构体字节对齐方式与C语言中的字节对齐方式相同。在C语言中,结构体字节对齐方式是根据结构体中最宽的成员变量的大小来确定的。具体来说,如果结构体中最宽的成员变量的大小为n,则结构体中的所有成员变量都必须按照n的倍数进行对齐。如果结构体中的某个成员变量的大小不是n的倍数,则需要在该成员变量后面填充一些字节,使得下一个成员变量能够按照n的倍数进行对齐。 在Python中,使用ctypes创建的结构体也遵循上述的字节对齐方式。具体来说,如果结构体中最宽的成员变量的大小为n,则结构体中的所有成员变量都必须按照n的倍数进行对齐。如果结构体中的某个成员变量的大小不是n的倍数,则需要在该成员变量后面填充一些字节,使得下一个成员变量能够按照n的倍数进行对齐。 下面是一个示例代码,演示了如何使用struct模块按照上述的格式解析结构体中的数据: ```python import ctypes import struct # 定义结构体 class MyStruct(ctypes.Structure): _fields_ = [ ("a", ctypes.c_int), ("b", ctypes.c_char), ("c", ctypes.c_double), ] # 假设从C代码中获取到的结构体数据是buf buf = b'\x01\x00\x00\x00\x41\x00\x00\x00\x00\x00\x00\x00' # 使用struct模块解析结构体中的数据 mystruct = MyStruct() struct_size = ctypes.sizeof(mystruct) mystruct_data = struct.unpack("i c d", buf[:struct_size]) mystruct.a, mystruct.b, mystruct.c = mystruct_data # 打印解析后的数据 print(mystruct.a) # 输出:1 print(mystruct.b) # 输出:b'A' print(mystruct.c) # 输出:0.0 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值