我对python ctypes模块并不陌生,但这是我在一个代码中首次尝试将C ++,C和Python结合起来。我的问题似乎与Seg fault when using ctypes with Python and C++非常相似,但我似乎无法以同样的方式解决问题。
我有一个名为Header.cpp
的简单C ++文件:
#include <iostream>
class Foo{
public:
int nbits;
Foo(int nb){nbits = nb;}
void bar(){ std::cout << nbits << std::endl; }
};
extern "C" {
Foo *Foo_new(int nbits){ return new Foo(nbits); }
void Foo_bar(Foo *foo){ foo->bar(); }
}
我使用:
编译到共享库g++ -c Header.cpp -fPIC -o Header.o
g++ -shared -fPIC -o libHeader.so Header.o
和一个名为test.py
的简单Python包装器:
import ctypes as C
lib = C.CDLL('./libHeader.so')
class Foo(object):
def __init__(self,nbits):
self.nbits = C.c_int(nbits)
self.obj = lib.Foo_new(self.nbits)
def bar(self):
lib.Foo_bar(self.obj)
def main():
f = Foo(32)
f.bar()
if __name__ == "__main__":
main()
我希望当我调用test.py时,我应该将32号打印到屏幕上。但是,我得到的只是一个分段错误。如果我更改构造函数以在堆栈上返回类实例(即没有new
调用)然后传递对象,程序将按预期执行。此外,如果我更改bar
类中的Foo
方法,使其不使用nbits
成员,则程序不会出错。
我对C ++的理解有限,但事实上我可以在C和C ++中使用这个函数,但在Python中却没有,这有点令人困惑。任何帮助将不胜感激。
更新:感谢下面的评论之一,问题已经解决。在这种情况下,需要明确声明C函数的restype和argtypes。即以下内容被添加到代码中:
lib.Foo_new.restype = C.c_void_p;
lib.Foo_new.argtypes = [C.c_int32];
lib.Foo_bar.restype = None;
lib.Foo_bar.argtypes = [C.c_void_p];
答案 0 :(得分:0)
我会尝试以下方法:
extern "C"
{
Foo *Foo_new(int nbits)
{
Foo *foo = new Foo(nbits);
printf("Foo_new(%d) => foo=%p\n", nbits, foo);
return foo;
}
void Foo_bar(Foo *foo)
{
printf("Foo_bar => foo=%p\n", foo);
foo->bar();
}
}
查看foo
的值是否匹配。
此外,您可能希望查看Boost.Python以简化创建C ++对象的Python绑定。