PYTHON:将字节数组转换为ctype Struct

2022-04-15 00:00:00 python ctypes bytearray

问题描述

我有一个‘n’字节数组。这与定义的ctype之一相同。结构。我想将这个字节数组类型转换为这个结构。这样我就可以访问这里的每一个成员。我如何才能做到这一点?

class ABC(Structure):
    _fields_ = [("a", c_uint), ("b", c_ushort), ("c", c_ushort)]

class DEF(Structure):
    _fields_ = [("abc", ABC), ("i", I)]

b = bytearray(b'x88x08xc0xf9x02x85x10x00xcc')

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'x88x08xc0xf9x02x85x10x00xcc')

    # 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字节。 我还将c_uint8用于DEF.i字段,因为这可能就是您的意思(在您的示例中,I不是一个类型)。

输出:

abc.a: 0xf9c00888
abc.b: 0x8502
abc.c: 0x10
i: 0xcc

如果您希望值使用大端(而不是默认的小端),请使用ctypes.BigEndianStructure更改结构的端序。

相关文章