Использование перечислений в ctypes.Structure
Вопрос
У меня есть структура, к которой я обращаюсь через ctypes:
struct attrl {
char *name;
char *resource;
char *value;
struct attrl *next;
enum batch_op op;
};
На данный момент у меня есть код Python, например:
# struct attropl
class attropl(Structure):
pass
attrl._fields_ = [
("next", POINTER(attropl)),
("name", c_char_p),
("resource", c_char_p),
("value", c_char_p),
Но я не уверен, что использовать для batch_op
перечисление.Должен ли я просто сопоставить это с c_int
или ?
Решение
По крайней мере для GCC enum
это просто простой числовой тип.Это может быть 8-, 16-, 32-, 64-битный или любой другой (я тестировал его с 64-битными значениями), а также signed
или unsigned
.Я думаю, оно не может превысить long long int
, но на практике вам следует проверить дальность действия вашего enum
s и выберите что-то вроде c_uint
.
Вот пример.Программа С:
enum batch_op {
OP1 = 2,
OP2 = 3,
OP3 = -1,
};
struct attrl {
char *name;
struct attrl *next;
enum batch_op op;
};
void f(struct attrl *x) {
x->op = OP3;
}
и Python:
from ctypes import (Structure, c_char_p, c_uint, c_int,
POINTER, CDLL)
class AttrList(Structure): pass
AttrList._fields_ = [
('name', c_char_p),
('next', POINTER(AttrList)),
('op', c_int),
]
(OP1, OP2, OP3) = (2, 3, -1)
enum = CDLL('./libenum.so')
enum.f.argtypes = [POINTER(AttrList)]
enum.f.restype = None
a = AttrList(name=None, next=None, op=OP2)
assert a.op == OP2
enum.f(a)
assert a.op == OP3
Другие советы
С использованием c_int
или c_uint
было бы хорошо.В качестве альтернативы существует рецепт в кулинарной книге для класса перечисления.