3

我正在尝试使用 ctypes 在 Python 中包装一个 C 库。函数调用需要我使用文档实现的回调函数。问题是回调函数需要来自库的自定义对象。这是我试图在 python 中复制的 C 代码

void outputCallback(const A* a, void* b) {
    //
}

a = function1(0, 0, outputCallback, 0, 0)

头文件中A的结构体定义为:

typedef struct A
{
     const unsigned char* a1;
     unsigned int a2;
} A;

以及我对 Python 等价物的尝试。

class A(Structure):
    _fields_ = [
       ("a1", ?, ?),
       ("a2", c_int, 16)]

class Callback():
    def outputCallback(self, a):
        print a.a2
        return 1

cb = Callback() 

CMPFUNC = CFUNCTYPE(c_int, POINTER(A))

cb.cmp_func = CMPFUNC(cb.outputCallback)

cdll.LoadLibrary("library.so")
libc = CDLL("library.so")

a = libc.function1(0, 0, cb.cmp_func, 0, 0)

我将回调包装在一个类中的原因是因为这篇文章。基本上这是我试图阻止这个回调被垃圾收集。

谢谢你的尽心帮助。

4

1 回答 1

4

那里有一些错误,有些是基本的 Python 错误:

from ctypes import *

class A(Structure):
    _fields_ = [
       ("a1", c_char_p),
       ("a2", c_int)]


class Callback(object):
    def outputCallback(self, a, b):         # outputCallback(): ?
        print a.contents.a1, a.contents.a2
                                            # The prototype of the `outputCallback`
                                            # tells that it returns nothing, `void`
cb = Callback()
CMPFUNC     = CFUNCTYPE(None, POINTER(A), c_void_p) # so `restype` shoud be
                                                    # None, void

cb.cmp_func = CMPFUNC(cb.outputCallback)

libc = CDLL('library.so')
libc.function1(0, 0, cb.cmp_func, 0, 0)

测试DLL:

#include <stdio.h>

#ifdef _WIN32
#define DLLEXPORT __declspec(dllexport) 
#else
#define DLLEXPORT
#endif

#ifdef __cplusplus
extern "C" {
#endif

typedef struct A
{
     const unsigned char* a1;
     unsigned int a2;
} A;

int DLLEXPORT function1(int a,
                        int b,
                        void (*outputCallback)(const A* a, void* b),
                        int c,
                        int d)
{
    A obj;

    obj.a1 = "Hello";
    obj.a2 = 5;

    outputCallback(&obj, NULL);
    return 0;
}

#ifdef __cplusplus
};
#endif

测试:

>gcc library.c -o library.so -shared

>python py.py
Hello 5

>
于 2013-03-27T19:45:48.160 回答