ctypes / C ++ segfault访问成员变量

时间:2012-12-06 23:03:57

标签: c++ python ctypes

我对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]; 

1 个答案:

答案 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绑定。