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

标签 c++ python c pointers ctypes

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

相关文章:

c++ - 终止函数模板递归

python - pycharm python3 输入错误

c++ - 如何在 C/C++ 中做 TCP 连接池

C++ Switch 语句大小写错误

c++ - 如何在 QT Creator 中关闭 g++ 中的 unicode 标志?

python - 如何在Python中从另一列(连续数据)的范围数据中提取一列的平均值、最大值和最小值

python - 如何以root身份运行selenium chromedriver? (即使使用 --no-sandbox 也不起作用

从函数、指针读取输入时,cs50 Credit.c 停止工作?

c - C 中的数组很麻烦

c++ - Opencv 在 cv::Mat 中存储 Yuyv (YCrCb)