12

C 函数myfunc对更大的数据块进行操作。结果以块的形式返回给回调函数:

int myfunc(const char *data, int (*callback)(char *result, void *userdata), void *userdata);

使用ctypesmyfunc ,从 Python 代码调用并将结果返回给 Python 回调函数并不是什么大问题。这个回调工作正常。

myfunc = mylib.myfunc
myfunc.restype = c_int
myfuncFUNCTYPE = CFUNCTYPE(STRING, c_void_p)
myfunc.argtypes = [POINTER(c_char), callbackFUNCTYPE, c_void_p]

def mycb(result, userdata):
    print result
    return True

input="A large chunk of data."
myfunc(input, myfuncFUNCTYPE(mycb), 0)

但是,有没有办法将 Python 对象(比如列表)作为用户数据提供给回调函数?为了存储结果块,我想做例如:

def mycb(result, userdata):
    userdata.append(result)

userdata=[]

但我不知道如何将 Python 列表转换为 c_void_p,以便可以在调用 myfunc 时使用它。

我目前的解决方法是将链表实现为 ctypes 结构,这很麻烦。

4

1 回答 1

3

我想你可以使用Python C API来做到这一点......也许你可以使用 PyObject 指针。

编辑:正如操作在评论中指出的那样,py_objectctypes 中已经有一个现成的类型,所以解决方案是首先ctypes.py_object从 python 列表中创建一个对象,然后将其转换c_void_p为将其作为参数传递给 C 函数(我认为这一步可能是不必要的,因为void*应该接受任何指针类型的参数,并且只传递一个 ) 会更快byref。在回调中,完成了相反的步骤(从 void 指针转换为指向的指针py_object,然后获取内容的值)。

一种解决方法可能是为您的回调函数使用闭包,以便它已经知道它必须在哪个列表中附加项目......

myfunc = mylib.myfunc
myfunc.restype = c_int
myfuncFUNCTYPE = CFUNCTYPE(STRING)
myfunc.argtypes = [POINTER(c_char), callbackFUNCTYPE]


def mycb(result, userdata):
    userdata.append(result)

input="A large chunk of data."
userdata = []
myfunc(input, myfuncFUNCTYPE(lambda x: mycb(x, userdata)))
于 2010-03-18T13:16:55.793 回答