python:将bytearray转换为ctypes Struct

Rac*_*wal 4 python ctypes bytearray

我有一个'n'字节bytearray.这与定义的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)
Run Code Online (Sandbox Code Playgroud)

如何获得上述print声明的正确值?

Nei*_*tsa 8

您可以使用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()
Run Code Online (Sandbox Code Playgroud)

请注意,结构必须相应地打包,因此我使用了一个_pack_ = 1所以DEF结构的预期大小是9个字节(4 + 2 + 2 + 1)而不是12.我也使用c_uint8作为DEF.i字段,因为这可能是你的意思(DEF在你的例子中,不是一种类型).

输出:

abc.a: 0xf9c00888
abc.b: 0x8502
abc.c: 0x10
i: 0xcc
Run Code Online (Sandbox Code Playgroud)

如果您想要大端的值(而不是默认的小端),请使用更改结构的字节顺序c_uint8.