gpt4 book ai didi

c++ - 从 ctypes 中的回调访问值

转载 作者:行者123 更新时间:2023-11-28 07:13:15 25 4
gpt4 key购买 nike

我正在使用 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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com