c++ - 使用 Pybind11 将 Eigen::Tensor 暴露给 Python
问题描述
我正在尝试使用 pybind11 将特征张量暴露给 python。我可以毫无问题地编译所有内容,并且可以成功地将其导入 python。但是,无法将数据转换为 python 类型。我尝试了两种方法。一是直接暴露数据,二是使用映射。两者都在 python 环境中失败。
#include <pybind11/pybind11.h>
#include <pybind11/stl.h>
#include <pybind11/stl_bind.h>
#include <pybind11/numpy.h>
#include <pybind11/eigen.h>
#include <unsupported/Eigen/CXX11/Tensor>
class myclass{
myclass(){
m_data = new float[m_dim1*m_dim2*m_dim3]; // Contiguous data that represents a three dimensional array
for(int i = 0; i<m_dim1*m_dim2*m_dim3; i++)
m_data[i] = i;
m_tensor = Eigen::TensorMap<Eigen::Tensor<float, 3>>(m_data, m_dim1, m_dim2, m_dim3);
}
Eigen::TensorMap<Eigen::Tensor<float, 3>>& getDataUsingMapping() { Eigen::TensorMap<Eigen::Tensor<float, 3>> temp(m_data, m_dim1, m_dim2, m_dim3); return temp; }
Eigen::Tensor<float, 3>& getDataWithoutUsingMapping() { return m_tensor};
private:
Eigen::Tensor<float, 3> m_tensor;
// In fact, m_data, m_dim1, m_dim2, m_dim3 all are
// read from a data file but for this example let's
// assume some values.
float* m_data;
int m_dim1 = 2, m_dim2 = 5, m_dim3 = 10;
}
PYBIND11_MODULE(example, m) {
py::class_<myclass>(m, "myclass")
.def(py::init<>())
.def("getDataUsingMapping", &myClass::getDataUsingMapping, py::return_value_policy::reference)
.def("getDataWithoutUsingMapping", &myClass::getDataWithoutUsingMapping, py::return_value_policy::reference);
}
我希望能够用它的维度信息在 python 中处理这个 3D 数组(m_dim1, m_dim2, m_dim3)
。
这是我尝试在 python 中获取数据后收到的错误消息。
>>> import example
>>> d = example()
>>>
>>> DataInPython = d.getDataUsingMapping()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Unable to convert function return value to a Python type! The signature was
(self: example) -> Eigen::TensorMap<Eigen::Tensor<float,3,0,__int64>,0,Eigen::MakePointer>
>>>
>>>
>>> DataInPython = d.getDataWithoutUsingMapping()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Unable to convert function return value to a Python type! The signature was
(self: example) -> Eigen::Tensor<std::complex<float>,3,0,__int64>
Did you forget to `#include <pybind11/stl.h>`? Or <pybind11/complex.h>,
<pybind11/functional.h>, <pybind11/chrono.h>, etc. Some automatic
conversions are optional and require extra headers to be included
when compiling your pybind11 module.
我尝试包含所有 pybdin11 包含文件,但没有解决问题。有人可以帮助我吗?
解决方案
该 C++ 代码无法编译并且该 python 代码不可能像发布的那样运行,但是在修复这些并进行逻辑更改之后,结论仍然是 pybind11 不支持来自“unsupported/Eigen/CXX11/Tensor”的 TensorMap,因为类不提供与其他 Eigen 映射类相同的接口。
我本来希望映射器脚轮的专业化能够自动工作,但要明确地这样做:
template<>
struct py::detail::type_caster<Eigen::TensorMap<Eigen::Tensor<float, 3>>, void>
: py::detail::eigen_map_caster<Eigen::TensorMap<Eigen::Tensor<float, 3>>> {};
pybind11::detail::EigenProps 的实例化失败,b/c TensorMap 不提供其维度与 cols/rows/stride。因此,SFINAE 阻止了脚轮的自动生成。
除了使用名为“不支持”的目录中的标头之外,没有其他选择吗?如果没有,最好的办法是将 TensorMap 的内容复制到一个 numpy 数组中,并以自定义的形式返回getDataUsingMapping
:有几个示例说明如何在 SO 上执行此操作,无论是否复制。(除非您愿意展平张量,否则 EigenProps 的专门化将不起作用,但您可以将其用作示例来为 TensorMap 编写新的泛型类型转换。)
推荐阅读
- vb.net - 如何使用控件在组框中验证文本框
- python - 如何在 Python 中查找所有可能的字典值组合,这些组合在 Python 中加起来为某个数字,同时仍保留键名
- heroku - 如何使用 Next.js 和 Heroku 管理错误 R14(超出内存配额)?
- sql - 嵌套 where 语句
- ruby-on-rails - 如何使用 ActiveSupport::Notifications 订阅所有查询方法
- javascript - Webpack 5 资产模块与 woff 文件有关
- node.js - 异步/等待的nodejs超时
- reactjs - react.js 和 usemutation 的无限循环
- android - 多线程 runOnUiThread
- json - 在一个混合数据类型的系列中,如何将偶尔出现的列表和字典转换成字符串?