Ctypes:获取指向struct字段的指针

时间:2018-05-25 14:11:11

标签: python pointers struct ctypes

我需要一个带有struct字段地址的指针。 重要提示:我正在为一组c结构构建一个序列化程序,所以我想迭代结构的字段并获取每个结构的地址作为指针。 我知道有一种方法可以使用它们的字段对象和偏移属性,它给出了结构本身地址的偏移量,但它是一个通用指针。 你能告诉我一个关于如何迭代结构字段的方法,并为每个字段获取一个具有正确内部类型的ctypes指针吗?

1 个答案:

答案 0 :(得分:2)

这是一种方式......

类变量Test._fields_定义属性及其C类型。通过分配字段生成的类属性包含有关该属性的偏移量的信息,例如:

>>> Test.a
<Field type=c_long, ofs=0, size=4>
>>> Test.b
<Field type=c_double, ofs=8, size=8>
>>> Test.c
<Field type=c_char_Array_10, ofs=16, size=10>

对于每种类型,from_buffer()方法使用与现有类型相同的数据缓冲区构建C类型。结构的一个实例实现了访问其数据所需的缓冲区API,因此,如果您知道结构元素数据的偏移量,则可以生成引用相同数据的ctypes类型,并创建指向它的指针。

from ctypes import *

class Test(Structure):
    _fields_ = [('a',c_int),
                ('b',c_double),
                ('c',c_char * 10)]

x = Test(1,2.5,b'abc')

for field,ftype in Test._fields_:

    # Look up each attribute in the class, and get its offset.
    ofs = getattr(Test,field).offset

    # Create a pointer to the same field type using the same data.
    p = pointer(ftype.from_buffer(x,ofs))
    print(p,p.contents)

输出:

<__main__.LP_c_long object at 0x000001932DE0EEC8> c_long(1)
<__main__.LP_c_double object at 0x000001932DE0ECC8> c_double(2.5)
<__main__.LP_c_char_Array_10 object at 0x000001932DE0EDC8> <__main__.c_char_Array_10 object at 0x000001932DE0ECC8>