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
其中 SheetData
是 typedef
用于:
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::list
的 std::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
问题
我正在尝试转换由在 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
其中 SheetData
是 typedef
用于:
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::list
的 std::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