(boost.python)暴露重载的operator +()时出错。 “TypeError:找不到to_python(按值)转换器”

时间:2011-10-11 15:19:07

标签: c++ python boost boost-python

我是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_policyto_python_value,这样我可以避免这种繁琐的运营商重写?

1 个答案:

答案 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> >())