0

我正在尝试使用 Cython 将我的 c++ 类包装到 python 中。我能够创建so文件并在python中导入,但是我的c++类的构造函数被多次调用。

这是我到目前为止所做的。
文件:DBReader.h

#ifndef _DB_READER_H_
#define _DB_READER_H_

#include <iostream>

class DBReader
{
  public:
    DBReader();
    ~DBReader();
};

#endif /* _DB_READER_H_ */

文件:DBReader.cxx

#include "DBReader.h"

DBReader::DBReader()
{
  std::cout << "Hello\n";
}


DBReader::~DBReader()
{

}

文件:dbreader.pyx

# distutils: language = c++
# distutils: sources = DBReader.cxx

cdef extern from "DBReader.h":
    cdef cppclass DBReader:
        DBReader() except +

cdef class PyDBReader:
    cdef DBReader c_dbreader      # hold a C++ instance which we're wrapping
    def __cinit__(self):
        self.c_dbreader = DBReader()

文件:setup.py

from distutils.core import setup, Extension
from Cython.Build import cythonize
from Cython.Distutils import build_ext

modules = [Extension("dbreader",
                sources=["dbreader.pyx", "../DBReader.cxx"],
                include_dirs = [".."],
                libraries = ["xxxx"],
                library_dirs = ["/usr/local/lib64"],
                language = "c++")]

setup(ext_modules = modules, cmdclass = {"build_ext" : build_ext})

运行后setup.py build_ext --inplace创建 dbreader.so 文件。而且我可以在python中导入。但正如您在下面看到的:Hello 被打印三次。

>>> import dbreader
>>> h = dbreader.PyDBReader()
Hello
Hello
Hello
>>>
4

1 回答 1

1

这一行的问题:

self.c_nodegraph_dbreader = RtlNodeGraphDBReader()

如果你只想要一个默认构造的实例,你已经得到了,没有这条线。

通过添加这一行,您明确地构建了第二个这样的实例,然后将其复制到原始实例上,并且……我不确定这一切是如何工作的,但我的猜测是自动生成的复制构造函数正在调用第三次使用默认构造函数。

于 2018-03-25T08:00:50.623 回答