pybind11:Python 到 C++ 数据类型的转换不起作用

pybind11: Python to C++ Datatype Conversion Not Working

问题

我正在尝试转换由在 C++ 代码中调用的 python 函数返回的列表列表。虽然 pybind11 库允许从 python 数据类型到 C++ 数据类型的类型转换,但我尝试将 python 返回的列表列表转换为 std::list of std::list of C++ strings, 每次都失败。

代码

这里是python函数(函数returns一个包含字符串值的列表列表):

def return_sheet(self):

     """Returns the sheet in a list of lists

     """

     dataTable = []

     for r in range(self._isheet.nrows):

         datalist = []

         for c in range(self._isheet.ncols):

             datalist.append(self._isheet.cell_value(r,c))

         dataTable.append(datalist)

 return dataTable

我在这里使用 pybind11:

在 C++ 中调用它
py::list obj = _tool.attr("return_sheet")();

data = py::cast<SheetData>(obj); // This is where the problem lies, This cast crashes the program

其中 SheetDatatypedef 用于:

typedef std::list<std::list<std::string> > SheetData;

在调试时,我发现程序实际上是在这一行崩溃:

py::object dataTable = _tool.attr("return_sheet")(); // Where _tool.attr("return_sheet")() gives an py::object which is a list of list of str

有人知道吗,我怎样才能成功地将 python 的列表列表转换为 C++ 的 std::liststd::list

编辑

这是我在 c++ [xlanalyser.py] 中嵌入的 python 程序文件: https://pastebin.com/gARnkMTv

这里是 C++ 代码 [main.cpp] : https://pastebin.com/wDDUB1s4

注意:xlanalyser.py 中的所有其他函数在嵌入到 c++ 时不会导致崩溃 [只有 return_sheet() 函数导致崩溃]

您可以使用 Python/C API 作为解决方法(检查函数 CastToSheetData)。我在下面包含了完整的示例:

program.py

def return_matrix():

    dataTable = []
    for r in range(0,2):
        datalist = []

        for c in range(0,2):
            datalist.append(str(r+c))

        dataTable.append(datalist)
    return dataTable

main.cpp

#include <pybind11/embed.h>
#include <iostream>
#include <list>
#include <string>

typedef std::list<std::list<std::string> > SheetData;

namespace py = pybind11;

SheetData CastToSheetData(PyObject *obj)
{
    SheetData data;
    PyObject *iter = PyObject_GetIter(obj);

    if (!iter)
        return data;
    while (true) {
        std::list<std::string> aux_list;
        PyObject *next = PyIter_Next(iter);
        if (!next) {
            // nothing left in the iterator
            break;
        }
        PyObject *iter2 = PyObject_GetIter(next);
        if (!iter2)
            continue;
        while(true) {
            PyObject *next2 = PyIter_Next(iter2);
            if (!next2) {
                // nothing left in the iterator
                break;
            }
            PyObject* pyStrObj = PyUnicode_AsUTF8String(next2); 
            char* zStr = PyBytes_AsString(pyStrObj); 
            std::string foo(strdup(zStr));
            aux_list.push_back(foo);
            Py_DECREF(pyStrObj);
        }
        data.push_back(aux_list);
    }

    return data;
}


int main()
{
    py::scoped_interpreter guard{};
    py::module calc = py::module::import("program");
    py::object result = calc.attr("return_matrix")();

    SheetData data = CastToSheetData(result.ptr());

    for (auto l : data)
    {
        std::cout << "[ ";
        for(auto s : l)
            std::cout << s <<  " ";
        std::cout << "]" << std::endl;
    }

    return 0;
}

输出:

[ 0 1 ]
[ 1 2 ]

可能,最好的方法是使用 load 方法

中的 CastToSheetData 函数中的代码来自定义 type_caster