我是boost.python的新手,我收到了这个错误,我想得到一些帮助。作为一个更大的项目的一部分,我正在为我的矢量类编写一个包装器。正如您将从下面的代码中注意到的,这个类可以是2D或3D,但对于我当前遇到的问题并没有太大的区别。我试图包装我在矢量类中定义的一些运算符,只要函数(或重载运算符)不返回Vector类型的“值”,这样工作正常,所以我猜python不知道是什么处理我按值返回的对象。这是代码:
struct D2
{
//...
typedef Eigen::Matrix< long double, 2, 1 > Vector;
//...
};
struct D3
{
//...
typedef Eigen::Matrix< long double, 3, 1 > Vector;
//...
};
现在定义了以下宏来为2个可能的维度执行相同的工作
BOOST_PYTHON_MODULE(types)
{
using namespace boost::python;
#define GPS_PY_EXPOSE_VECTOR(DIM, NAME) \
class_<DIM::Vector>(NAME) \
.def(init<DIM::Vector>()) \
.def("norm", &DIM::Vector::norm) \
.def("__setitem__", &PyVectorHelper<DIM>::setVectorElem) \
.def("__getitem__", &PyVectorHelper<DIM>::getVectorElem) \
.def(self += self) \
.def(self -= self) \
.def(self + self) /*Not working!!!*/ \
.def(self - self) /*Not working!!!*/ \
.def("toStr", &PyVectorHelper<DIM>::toPyString) \
.def("__str__", &PyVectorHelper<DIM>::toStdString) \
;
GPS_PY_EXPOSE_VECTOR(D2, "Vector2D")
GPS_PY_EXPOSE_VECTOR(D3, "Vector3D")
}
上面的代码工作正常,除非我尝试在python中使用“+”或“ - ”运算符,如下所示:
>>>import types
>>>v1 = types.Vector2D()
>>>v2 = types.Vector2D()
>>>v3 = v1 + v2 //ERROR!!!
错误如下:
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: No to_python (by-value) converter found for C++ type: Eigen::CwiseBinaryOp<Eigen::internal::scalar_sum_op<long double>, Eigen::Matrix<long double, 2, 1, 0, 2, 1> const, Eigen::Matrix<long double, 2, 1, 0, 2, 1> const>
我按照教程here,发现没有与互联网上的错误相关的信息。
非常欢迎任何帮助! 提前谢谢。
在Luc Danton的回答后编辑
致Luc Danton:谢谢你的回答,我试过你所说的,但它不起作用。这是我得到的错误:
error: no matching function for call to ‘boost::python::class_<Eigen::Matrix<long double, 2, 1, 0, 2, 1>, boost::python::detail::not_specified, boost::python::detail::not_specified, boost::python::detail::not_specified>::def(boost::python::detail::operator_<(boost::python::detail::operator_id)0u, boost::python::self_ns::self_t, boost::python::self_ns::self_t>, boost::python::return_value_policy<boost::python::to_python_value<Eigen::Matrix<long double, 2, 1, 0, 2, 1> >, boost::python::default_call_policies>)’
然而,由于你的评论,我得到了这个想法:编写一个辅助函数来执行加法并转换特征“表达式对象”(这是表示运算符的私有类型)。这是:
template <typename TDim>
struct PyVectorHelper
{
//...
static typename TDim::Vector sum(typename TDim::Vector const &v1, typename TDim::Vector const &v2)
{
return v1 + v2;
}
//...
};
然后
.def("__add__", &PyVectorHelper<DIM>::sum)
这样可以正常使用,因为PyVectorHelper<DIM>::sum
会返回DIM::Vector
。问题是几乎每个被Eigen重载的运算符都返回一个表达式对象,这意味着我必须为几乎所有这些函数实现类似的函数!这不好也不实用:P。
也许仍然可以使用return_value_policy
和to_python_value
,这样我可以避免这种繁琐的运营商重写?
答案 0 :(得分:3)
矩阵/载体库似乎正在使用表达模板。在这种情况下,正如您所说,operator+
不返回向量,而是返回表示操作的私有类型。返回的值被遗留在C ++方面,由于私有类型未注册,因此无法传递给Python。
现在我对Boost.Python不是很熟悉,但我认为你可以用return value policy解决你的问题。传递return_value_policy<to_python_value<DIM::Vector> >
作为策略会强制将返回的类型转换为您已注册的DIM::Vector
。这看起来像是:
.def(self + self, return_value_policy<to_python_value<DIM::Vector> >())