如何使用ctypes从C++函数返回对象?

lel*_*man 4 c++ python ctypes

我有两个C++类,Container和Item,看起来像:

class Item{
public:
    Item(std::string* name,int id);
    std::string* getName();
private:
    std::string* name;
    int id;
};

class Container {
public:
    Container();
    Item* getItem(int id);
private:
    std::vector<Item*> items;
};
Run Code Online (Sandbox Code Playgroud)

我想在Python中创建和使用Container,所以我编写了一个C接口来编译共享库:

extern "C" {
    Container* Container_init(){return new Container();}
    Item* Container_getItem(Container* container,int id){return container->getItem(id);}
    std::string* Item_getName(Item* item){return item->getName();}
}
Run Code Online (Sandbox Code Playgroud)

和一个Python包装器:

from ctypes import *

lib = cdll.LoadLibrary(myLibPath)

class Item(object):
    def getName(self):
        return lib.Item_getName(self.obj)

class Container(object):
    def __init__(self):
        self.obj = lib.Container_init()

    def getItem(self,id):
        return lib.Container_getItem(self.obj,id)


lib.Container_getItem.restype = Item
lib.Container_getItem.argtypes = [c_void_p,c_int]

c = Container()
print c.getItem(5).getName()
Run Code Online (Sandbox Code Playgroud)

当这段代码运行时,它会引发一个TypeError"object()不带参数"

return lib.Container_getItem(self.obj,id)
Run Code Online (Sandbox Code Playgroud)

我在文档中读到了restype和argtype,但我显然遗漏了一些东西,如何Container.getItem在Python中返回一个Item?

fal*_*tru 5

替换Item_getName为以下char *而不是string *:

const char* Item_getName(Item* item) { return item->getName()->c_str(); }
Run Code Online (Sandbox Code Playgroud)

Item班级不见了__init__.改变如下(这是原因TypeError):

class Item(object):
    def __init__(self, obj):
        self.obj = obj
    def getName(self):
        return lib.Item_getName(self.obj)
Run Code Online (Sandbox Code Playgroud)

并在Python脚本中添加以下内容(在调用getName方法之前)以正确获取名称:

lib.Item_getName.restype = c_char_p
lib.Item_getName.argtypes = ()
Run Code Online (Sandbox Code Playgroud)

然后,你会得到你想要的.