Python ctypes - Ошибка доступа к строке данных в Structure .value - PullRequest
4 голосов
/ 14 сентября 2010

Я могу получить структуру, заполненную в результате dll-функции (как кажется, просматривая ее, используя x=buffer(MyData), а затем repr(str(buffer(x))).

Но возникает ошибка, если я пытаюсьчтобы получить доступ к элементам структуры, используя .value.

У меня есть VarDefs.h, который требует структуру, подобную этой:

typedef struct
{
  char Var1[8+1];
  char Var2[11+1];
  char Var3[3+1];
...
}TMyData

, которая должна быть передана в функцию, подобную этой:

__declspec(dllexport) int AFunction(TOtherData *OtherData, TMyData *MyData);

В Python я теперь могу объявить структуру таким образом (спасибо г-ну Мартелли: см. Здесь Python ctypes - сбой при приеме структур dll-функцией ):

class TMyData( Structure ):

    _fields_ = [
        ("Var1" , type( create_string_buffer(9) ) ),
        ("Var2" , type( create_string_buffer(12)) ),
...

Я вызываю функцию следующим образом: result = Afunction( byref(OtherData) , byref(MyData ) )

Как я уже сказал, когда я пытаюсь получить доступ к MyData.Var1.value, я получаю сообщение об ошибке (извините, сейчас не могу быть более конкретным!), Ноrepr(str(x)) где x - копия buffer(MyData) показывает, что в нем есть данные!

Как мне вместо этого сделать это? Спасибо!

Ответы [ 2 ]

3 голосов
/ 19 сентября 2010

Просто используйте print MyData.Var1.Массив символов преобразуется в строковый тип Python при доступе через экземпляр Structure, который не имеет метода .value.

Придуманный, рабочий пример:

Код DLLxc, скомпилированный с MSVC с "cl / LD xc")

#include <string.h>

typedef struct
{
    char Var1[5];
    char Var2[10];
    char Var3[15];
} TMyData;

__declspec(dllexport) int AFunction(TMyData *MyData)
{
    strcpy(MyData->Var1,"four");
    strcpy(MyData->Var2,"--nine---");
    strcpy(MyData->Var3,"---fourteen---");
    return 3;
}

Код Python

import ctypes as c

class TMyData(c.Structure):
   _fields_ = [
        ("Var1", c.c_char * 5),
        ("Var2", c.c_char * 10),
        ("Var3", c.c_char * 15)]

lib = c.CDLL('x')
data = TMyData()
lib.AFunction(c.byref(data))

print data.Var1
print data.Var2
print data.Var3
print data.Var1.value # error!

Выход

four
--nine---
---fourteen---
Traceback (most recent call last):
  File "C:\Python26\Lib\site-packages\Pythonwin\pywin\framework\scriptutils.py", line 436, in ImportFile
    my_reload(sys.modules[modName])
  File "C:\x.py", line 12, in <module>
    print data.Var1.value
AttributeError: 'str' object has no attribute 'value'
3 голосов
/ 14 сентября 2010

Структура, с которой вы пытаетесь использовать ctypes для взаимодействия, содержит несколько «массивов символов», а не «указателей на массивы символов». Вместо использования create_string_buffer(9) вам нужно использовать ctypes.c_char * 9.

class TMyData( ctypes.Structure ):
   _fields_ = [ ("Var1", ctypes.c_char * 9),
                ("Var2", ctypes.c_char * 12), ... ]
...