在ctypes中从回调中访问值

时间:2013-12-17 17:12:28

标签: c++ python c pointers ctypes

我正在使用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代码错误时。

1 个答案:

答案 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