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

ctypes/C++ segfault accessing member variables

本文关键字:成员 变量 访问 C++segfault ctypes      更新时间:2023-10-16

我对python ctypes模块并不陌生,但这是我第一次尝试将C++、C和python组合在一个代码中。在Python和C++中使用ctypes时,我的问题似乎与Seg故障非常相似,但我似乎无法用同样的方式解决问题。

我有一个简单的C++文件,名为Header.cpp:

#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调用),然后传递对象,程序将按预期执行。此外,如果我更改Foo类中的bar方法,使其不使用nbits成员,则程序不会出错。

我对C++的理解有限,但事实上,我可以在C和C++中按预期生成这个函数,但不能在Python中生成,这有点令人困惑。如有任何帮助,我们将不胜感激。

更新:由于下面的一条评论,问题已经解决。在这种情况下,需要为C函数显式声明resttype和argtype。即在python代码中添加了以下内容:

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]

我会尝试以下操作:

extern "C"
{
Foo *Foo_new(int nbits)
{
Foo *foo = new Foo(nbits);
printf("Foo_new(%d) => foo=%pn", nbits, foo);
return foo;
}
void Foo_bar(Foo *foo)
{
printf("Foo_bar => foo=%pn", foo);
foo->bar();
}
}

以查看CCD_ 7的值是否匹配。

此外,您可能希望查看Boost.Python以简化创建C++对象的Python绑定。