近日在作一组声纹聚类时,使用了另外一团队同窗开发的声纹距离算法。该算法对外提供的是一组so包,须要使用方本身去使用。在python中调用纯so包通常使用ctypes类库,用起来看起来简单但也有很多细节容易犯错。本次使用过程当中,就遇到传参的问题。python
目标so库中对外export的函数是大体以下的三个函数:算法
void* create_handler(); int extract_feature(void* hander); bool destroy(void* handler);
这三个函数使用起来倒也简单,顺序使用就能够了。但发现写成以下形式的python代码后,执行会直接segment fault。函数
import sys import ctypes so = ctypes.CDLL("./lib/libbase.so") p = so.create_handler() feature = so.extract_feature(p) so.destroy(p)
这段代码中p是int类型,由void*自动转来,在ctyeps中这种转型自己是没问题的。segment fault发生在extract_feature函数调用中,问题应当出在参数上,回传的handler已经不是原来的pointer了,致使访问指针出错。指针
查阅ctypes的文档后,发现ctypes能够声明so库中函数的参数,返回类型。试了试,显式声明后问题获得了解决,证实咱们的猜测是对的,确实指针发生了变化。修改后代码以下:rest
import sys import ctypes so = ctypes.CDLL("./lib/libbase.so") so.create_handler.restype=ctypes.c_void_p so.extract_feature.argtypes=[ctypes.c_void_p] so.destroy.argtypes=[ctypes.c_void_p] p = so.create_handler() feature = so.extract_feature(p) so.destroy(p)
ctypes中传递指针类型参数须要显式声明c函数的参数,返回类型。