Python ctypes - 访问结构体中的数据字符串.value失败

4 投票
2 回答
5766 浏览
提问于 2025-04-16 04:06

我通过一个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中,我现在能够这样声明这个结构体(感谢Martelli先生:可以查看这里 Python ctypes - dll function accepting structures crashes):

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

只需要用 print MyData.Var1 就可以了。当你通过一个 Structure 实例来访问字符数组时,它会自动转换成 Python 的字符串类型,而这个结构体并没有 .value 这个方法。

下面是一个简单的示例:

DLL 代码 (x.c,使用 MSVC 编译,命令是 "cl /LD x.c")

#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

你想用ctypes来处理的结构里,有几个“字符数组”,而不是“字符数组的指针”。所以,你应该用 ctypes.c_char * 9,而不是 create_string_buffer(9)

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

撰写回答