我正在尝试使用ctypes为C库创建一个python包装器。该库具有需要指向要传递的结构的指针的函数,该函数用作将来调用的句柄。
此结构具有指向另一个内部结构的指针,该内部结构还具有指向其他结构的指针。
typedef struct varnam {
char *scheme_file;
char *suggestions_file;
struct varnam_internal *internal;
} varnam;
varnam_internal
结构有指向sqlite数据库的指针等等
struct varnam_internal
{
sqlite3 *db;
sqlite3 *known_words;
struct varray_t *r;
struct token *v;
...
}
我尝试根据this SO回答忽略varnam_internal结构。像
这样的东西class Varnam(Structure):
__fields__ = [("scheme_file",c_char_p),
("suggestions_file",c_char_p),("internal",c_void_p)]
但这似乎不起作用,因为我认为图书馆需要分配varnam_internal
才能正常运作。
我应该在python中实现所有依赖结构吗? ctypes是否适合包装这样的库?我已经读过像Cython这样的替代品,但是我没有Cython的经验,所以这可行吗?
答案 0 :(得分:1)
没有理由在ctypes中定义varnam_internal
结构,因为您不需要访问它。无论您是否定义结构,您正在调用的库都将分配它。无论你遇到什么问题,都不是因为你没有用ctypes定义结构。
确保您正确拨打varnam_init
。它使用指针指针作为参数,这意味着您不能直接使用Varnam
类。你会想做这样的事情:
from ctypes import *
class Varnam(Structure):
__fields__ = [("scheme_file",c_char_p),
("suggestions_file",c_char_p),
("internal",c_void_p)]
varnam_ptr = POINTER(Varnam)
libvarnam = cdll.LoadLibrary("libvarnam.so") # on Linux
# libvarnam = cdll.libvarnam # on Windows
varnam_init = libvarnam.varnam_init
varnam_init.argtypes = [c_char_p, POINTER(varnam_ptr), POINTER(c_char_p)]
def my_varnam_init(scheme_file):
handle = varnam_ptr()
msg = c_char_p()
r = varnam_init(scheme_file. handle.byref(), msg.byref())
if r != 0:
raise Exception(msg)
return handle
以上代码完全未经测试,但会向您展示如何调用varnam_init
。