Как получить дескриптор поля 'type' из структуры ctypes или поля Union - PullRequest
4 голосов
/ 19 мая 2011

У меня есть структура с различными полями типов данных.Я хотел бы пройтись по полям структуры, проверить тип данных и установить для поля соответствующее значение.

У меня есть доступ к размеру и смещению поля через атрибуты .size и .offsetполе.Как я могу получить атрибут 'type' поля?Использование type (value) не печатает тип данных ctypes для определенного поля.Если я напечатаю значение , тогда я вижу тип данных ctypes, но, похоже, нет атрибута для прямого доступа к нему.

Как я могу получить прямой доступ к дескриптору поля типа?

from ctypes import *

class A(Structure):
    _fields_ = [("one", c_long),
                ("two", c_char),
                ("three", c_byte)]

>>> A.one
<Field type=c_long, ofs=0, size=4>
>>> A.one.offset
0
>>> A.one.size
4
>>> type(A.one)
<class '_ctypes.CField'>

В идеале я хотел бы получить тип поля, аналогичный приведенному ниже фрагменту ...

>>> A.one.type
c_long

Ответы [ 2 ]

5 голосов
/ 19 мая 2011

Это не поддерживается в API ctypes. Когда создается Field repr <Field type=c_long ..>, имя извлекается из встроенного типа следующим образом:

name = ((PyTypeObject *)self->proto)->tp_name;

Для вашего поля член self->proto указывает на c_long, но я не нахожу места в Python 2.7 cfield.c, где вы можете получить значение self->proto. Вас могут заставить либо:

  1. Создайте свое собственное отображение из name -> type.
  2. (yuck) Разобрать repr для <Field type=X и использовать getattr(ctypes, X) для извлечения объекта типа.

Просто чтобы продолжить пример option (1) , вот декоратор класса, который создает для вас отображение типа, добавив _typeof(cls, fld) метод класса:

from ctypes import *

def typemap(cls):
    _types = dict((getattr(cls, t), v) for t, v in cls._fields_)
    setattr(cls, '_typeof', classmethod(lambda c, f: _types.get(f)))
    return cls

@typemap
class A(Structure):
    _fields_ = [("one", c_long),
                ("two", c_char),
                ("three", c_byte)]

print A._typeof(A.one), A._typeof(A.two), A._typeof(A.three)

Результат:

<class 'ctypes.c_long'> <class 'ctypes.c_char'> <class 'ctypes.c_byte'>
4 голосов
/ 20 мая 2011

Просто используйте список _fields_:

>>> for f,t in A._fields_:
...  a = getattr(A,f)
...  print a,a.offset,a.size,t
...
<Field type=c_long, ofs=0, size=4> 0 4 <class 'ctypes.c_long'>
<Field type=c_char, ofs=4, size=1> 4 1 <class 'ctypes.c_char'>
<Field type=c_byte, ofs=5, size=1> 5 1 <class 'ctypes.c_byte'>
...