python: 将 bytearray 转换为 ctypes 结构
python: Convert bytearray to ctypes Struct
我有一个 'n' Byte 字节数组。这与定义的 ctypes.Sturcture 之一相同。我想将这个 bytearray 类型转换为这个结构。这样我就可以访问每个成员。我该怎么做?
class ABC(Structure):
_fields_ = [("a", c_uint), ("b", c_ushort), ("c", c_ushort)]
class DEF(Structure):
_fields_ = [("abc", ABC), ("i", I)]
b = bytearray(b'\x88\x08\xc0\xf9\x02\x85\x10\x00\xcc')
s = DEF()
print(s.abc.a)
如何获得上述 print
语句的正确值?
您可以在所需类型(而不是对象实例)上使用 from_buffer
:
from ctypes import Structure, c_uint, c_ushort, c_uint8
class ABC(Structure):
_pack_ = 1
_fields_ = [("a", c_uint), ("b", c_ushort), ("c", c_ushort)]
class DEF(Structure):
_pack_ = 1
_fields_ = [("abc", ABC), ("i", c_uint8)]
def main():
b = bytearray(b'\x88\x08\xc0\xf9\x02\x85\x10\x00\xcc')
# check if bytearray can be applied to structure.
if len(b) < ctypes.sizeof(DEF):
print("error: bytearray is too short for DEF.")
return
s = DEF.from_buffer(b)
print("abc.a: {:#x}".format(s.abc.a))
print("abc.b: {:#x}".format(s.abc.b))
print("abc.c: {:#x}".format(s.abc.c))
print("i: {:#x}".format(s.i))
if __name__ == '__main__':
main()
请注意,结构必须相应地打包,因此我使用了 _pack_ = 1
,因此 DEF
结构的预期大小为 9 字节(4 + 2 + 2 + 1)而不是 12。
我还在 DEF.i
字段中使用了 c_uint8
,因为这可能就是您的意思(在您的示例中,I
不是类型)。
输出:
abc.a: 0xf9c00888
abc.b: 0x8502
abc.c: 0x10
i: 0xcc
如果您想要大端(而不是默认的小端)的值,请使用 ctypes.BigEndianStructure
更改结构的字节顺序。
我有一个 'n' Byte 字节数组。这与定义的 ctypes.Sturcture 之一相同。我想将这个 bytearray 类型转换为这个结构。这样我就可以访问每个成员。我该怎么做?
class ABC(Structure):
_fields_ = [("a", c_uint), ("b", c_ushort), ("c", c_ushort)]
class DEF(Structure):
_fields_ = [("abc", ABC), ("i", I)]
b = bytearray(b'\x88\x08\xc0\xf9\x02\x85\x10\x00\xcc')
s = DEF()
print(s.abc.a)
如何获得上述 print
语句的正确值?
您可以在所需类型(而不是对象实例)上使用 from_buffer
:
from ctypes import Structure, c_uint, c_ushort, c_uint8
class ABC(Structure):
_pack_ = 1
_fields_ = [("a", c_uint), ("b", c_ushort), ("c", c_ushort)]
class DEF(Structure):
_pack_ = 1
_fields_ = [("abc", ABC), ("i", c_uint8)]
def main():
b = bytearray(b'\x88\x08\xc0\xf9\x02\x85\x10\x00\xcc')
# check if bytearray can be applied to structure.
if len(b) < ctypes.sizeof(DEF):
print("error: bytearray is too short for DEF.")
return
s = DEF.from_buffer(b)
print("abc.a: {:#x}".format(s.abc.a))
print("abc.b: {:#x}".format(s.abc.b))
print("abc.c: {:#x}".format(s.abc.c))
print("i: {:#x}".format(s.i))
if __name__ == '__main__':
main()
请注意,结构必须相应地打包,因此我使用了 _pack_ = 1
,因此 DEF
结构的预期大小为 9 字节(4 + 2 + 2 + 1)而不是 12。
我还在 DEF.i
字段中使用了 c_uint8
,因为这可能就是您的意思(在您的示例中,I
不是类型)。
输出:
abc.a: 0xf9c00888
abc.b: 0x8502
abc.c: 0x10
i: 0xcc
如果您想要大端(而不是默认的小端)的值,请使用 ctypes.BigEndianStructure
更改结构的字节顺序。