如何在Cython中将函数指针传递给外部程序
问题内容:
我正在尝试为C程序编写包装程序,以便可以从Python调用它。我正在使用Cython来执行此操作。C函数将回调函数作为参数,但是此回调函数仅在python程序运行时才知道。我一直在搜索如何执行此操作,似乎没有简单的解决方案,但以下方法似乎有效:
#python.py
libc = cdll.LoadLibrary("myfunc.so") #Callback function is defined in myfunc.so
....
c_wrapper(libc.fun, ...)
。
#c_wrapper.pyx
cdef extern void mainfunction(void *F, ...) #The intial C function we are wrapping
ctypedef void (*myfuncptr) ()
def c_wrapper(f, ...) # our function pointer is passed to the wrapper as a Python object
cdef myfuncptr thisfunc
thisfunc = (<myfuncptr*><size_t>addressof(f))[0]
mainfunction(thisfunc, ...)
此方法适用于C和FORTRAN函数(我假设它将适用于大多数已编译的语言)和Python函数(使用C类型),但似乎有点尴尬。在Cython中,还有其他更简单的方法吗?
谢谢
编辑:我无法更改我想包装的C库
问题答案:
我想你是知道
这个
?
是否可以从C调用我的Python代码?
答:是的,很容易。请遵循Cython来源分发中的Demos / callback /中的示例
因此,知道您可以卸载主要功能,让我们采取另一种方法。我写了几个愚蠢的函数来测试这一点:
/* lib.c -> lib.so */
#include <stdio.h>
void fn1(void) {
puts("Called function 1");
}
void fn2(void) {
puts("Called function 2");
}
然后,执行回调的函数
/* main.c -> main.so */
typedef void (*callback)();
void mainfunction(void *F) {
((callback)F)();
}
可以直接从Python传递:
>>> from ctypes import cdll
>>> lib = cdll.LoadLibrary('./lib.so')
>>> main = cdll.LoadLibrary('./main.so')
>>> main.mainfunction(lib.fn1)
Called function 1
>>> main.mainfunction(lib.fn2)
Called function 2
现在,让我们包装一个Python函数:
>>> from ctypes import CFUNCTYPE
>>> def pyfn():
... print "Called the Python function"
...
>>> CWRAPPER = CFUNCTYPE(None)
>>> wrapped_py_func = CWRAPPER(pyfn)
>>> main.mainfunction(wrapped_py_func)
Called the Python function