我正在使用Python / ctypes编写基于商业DLL的应用程序。此DLL读取flatfile并通过struct
返回数据。相关的C struct
看起来像这样:
struct System{
unsigned short user;
unsigned short version;
};
struct Message {
unsigned short header;
unsigned short data;
unsigned int messageType;
};
DLL附带的API提供了一个指向具有以下原型的函数的指针。此函数循环遍历flatfile并调用下一个定义的Callback
:
typedef int (__stdcall *Process) (const char* filename, const char* base, unsigned int flags, int user, Callback stdcallback);
和Callback
原型(上面的第五个参数)定义为:
typedef int (__stdcall *Callback) (const System* Sys, const Message* Msg);
在我的python文件中:
from ctypes import *
# using WinDLL for stdcall
lib = WinDLL('CVO.dll')
class System(Structure):
_fields_ = [('user', c_ushort),
('version', c_ushort)]
class Message(Structure):
_fields_ = [('header', c_ushort),
('data', c_ushort),
('messageType', c_uint)]
PROTO = WINFUNCTYPE(c_int, POINTER(System), POINTER(Message))
def py_callbck(Sys, Msg):
print Msg._type_.messageType
# return 0 to read the next line in the flatfile
return 0
Callback = PROTO(py_callbck)
pProcess lib.Process
pProcess.argtypes = [c_char_p,c_char_p,c_uint,c_int,PROTO]
pProcess.restype = c_int
pProcess(c_char_p('flat.ccf'),c_char_p(None),c_uint(0),c_int(0),Callback)
我的问题在于访问DLL传递给Msg
结构的值。代码应该打印c_int
(0,1,2),而是返回Field
类型,如下所示:
<Field type=c_ulong, ofs=136, size=4>
我知道平面文件正在处理中,因为我的Python代码打印出了数百个<Field ...>
语句,它本应该这样做。
我的问题是如何访问DLL推送到Msg
Structure
而不是返回<Field ...>
语句的值?
注意我对ctypes比较新,而且C很少,所以如果你发现定义任何问题,请告诉我。例如,我知道返回和对象的ctypes.pointer
和返回新类型的ctypes.POINTER
之间存在差异。当我使用ctypes.pointer
代码错误时。
答案 0 :(得分:1)
对于相对较新的ctypes,你在这里做得非常好。指针的_type_
属性是对指向数据类型的引用。在这种情况下,它是Message
。类属性messageType
是实例使用的CField
数据描述符。
您想要做的是取消引用获取Message
实例的指针。您可以使用[0]
下标或contents
属性:
def py_callbck(Sys, Msg):
print Msg[0].messageType # Msg.contents.messageType
# return 0 to read the next line in the flatfile
return 0
请记住保留对Callback
的引用,以防止垃圾被收集。目前,您将其引用为全局。没关系。
关于调用pProcess
的方式,在定义argtypes
时,通常不需要为简单类型手动创建ctypes对象。您可以更简单地使用以下内容:
pProcess('flat.ccf', None, 0, 0, Callback)
但要注意函数是否需要可写字符串缓冲区。在这种情况下,使用create_string_buffer
。这里没有必要,因为char *
参数都是const
。