我正在使用 Python/ctypes 编写基于商业 DLL 的应用程序。此 DLL 读取平面文件并通过 struct
返回数据秒。相关C struct
看起来像这样:
struct System{
unsigned short user;
unsigned short version;
};
struct Message {
unsigned short header;
unsigned short data;
unsigned int messageType;
};
DLL 附带的 API 提供了指向具有以下原型(prototype)的函数的指针。此函数循环遍历平面文件并调用 Callback
接下来定义:
typedef int (__stdcall *Process) (const char* filename, const char* base, unsigned int flags, int user, Callback stdcallback);
和 Callback
原型(prototype)(上面的第五个参数)定义为:
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
代码出错。
最佳答案
作为 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
。
关于c++ - 从 ctypes 中的回调访问值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20640497/