C++ 自定义 class 继承本征初始化列表
C++ Custom class inhereting Eigen initializer list
我正在尝试像这样从 Eigen 的向量 3 继承自定义 Vector3。
using Vector3 = Eigen::Vector3d;
enum class CustomEnum
{
E1, E2 E3, E4,
};
class CustomVector3 : public Vector3
{
public:
// Taken from here: https://eigen.tuxfamily.org/dox/TopicCustomizing_InheritingMatrix.html
CustomVector3():Vector3(){}
template<typename OtherDerived>
CustomVector3(const Eigen::MatrixBase<OtherDerived>& other) : Vector3(other) {}
// Initializer Constructor
template<typename OtherDerived>
CustomVector3(const std::initializer_list<OtherDerived>& other) : Vector3(other) {}
template<typename OtherDerived>
CustomVector3& operator=(const Eigen::MatrixBase<OtherDerived>& other){
this->Vector3::operator=(other);
return *this;
}
CustomEnum e = CustomEnum::E1;
};
目标是执行以下操作:
TEST_CASE("Initalizer List Test")
{
CustomVector3 vec = {1.1, 2.2, 3.3};
CHECK(vec.x() == 1.1);
CHECK(vec.y() == 2.2);
CHECK(vec.z() == 3.3);
assert(vec.frame == CustomEnum::E1);
}
没有初始化构造函数,我可以执行以下操作,但不是我上面提到的目标:
Vector3 data = {1.1, 2.2, 3.3};
CustomVector3 framedData = data;
使用初始化构造函数,我得到以下编译错误消息(我在我的代码中将其称为 FramedVector3,但为了清楚起见,我在此 post 中将其称为 CustomVector3):
/usr/include/eigen3/Eigen/src/Core/Matrix.h: In instantiation of ‘Eigen::Matrix<_Scalar, _Rows, _Cols, _Options, _MaxRows, _MaxCols>::Matrix(const T&) [with T = std::initializer_list<double>; _Scalar = double; int _Rows = 3; int _Cols = 1; int _Options = 0; int _MaxRows = 3; int _MaxCols = 1]’:
/home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:23:81: required from ‘FramedVector3::FramedVector3(const std::initializer_list<_Tp>&) [with OtherDerived = double]’
/home/seedship/uavAP/include/uavAP/Core/SensorData.h:128:38: required from here
/usr/include/eigen3/Eigen/src/Core/Matrix.h:294:31: error: no matching function for call to ‘Eigen::Matrix<double, 3, 1>::_init1<std::initializer_list<double> >(const std::initializer_list<double>&)’
294 | Base::template _init1<T>(x);
| ~~~~~~~~~~~~~~~~~~~~~~~~^~~
In file included from /usr/include/eigen3/Eigen/Core:462,
from /usr/include/eigen3/Eigen/Dense:1,
from /usr/local/include/cpsCore/Utilities/LinearAlgebra.h:13,
from /home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:9,
from /home/seedship/uavAP/include/uavAP/Core/SensorData.h:11,
from /home/seedship/uavAP/include/uavAP/Core/Orientation/ENU.h:8,
from /home/seedship/uavAP/src/Core/Orientation/ENU.cpp:5:
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:774:30: note: candidate: ‘void Eigen::PlainObjectBase<Derived>::_init1(Eigen::Index, typename Eigen::internal::enable_if<(((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime != 1) || (! Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value)) && ((! Eigen::internal::is_same<typename Eigen::internal::traits<T>::XprKind, Eigen::ArrayXpr>::value) || (typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == Eigen::Dynamic))), T>::type*) [with T = std::initializer_list<double>; Derived = Eigen::Matrix<double, 3, 1>; Eigen::Index = long int; typename Eigen::internal::enable_if<(((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime != 1) || (! Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value)) && ((! Eigen::internal::is_same<typename Eigen::internal::traits<T>::XprKind, Eigen::ArrayXpr>::value) || (typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == Eigen::Dynamic))), T>::type = std::initializer_list<double>]’
774 | EIGEN_STRONG_INLINE void _init1(Index size, typename internal::enable_if< (Base::SizeAtCompileTime!=1 || !internal::is_convertible<T, Scalar>::value)
| ^~~~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:774:43: note: no known conversion for argument 1 from ‘const std::initializer_list<double>’ to ‘Eigen::Index’ {aka ‘long int’}
774 | EIGEN_STRONG_INLINE void _init1(Index size, typename internal::enable_if< (Base::SizeAtCompileTime!=1 || !internal::is_convertible<T, Scalar>::value)
| ~~~~~~^~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: note: candidate: ‘template<class T> void Eigen::PlainObjectBase<Derived>::_init1(const Scalar&, typename Eigen::internal::enable_if<((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == 1) && Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value), T>::type*) [with T = T; Derived = Eigen::Matrix<double, 3, 1>]’
788 | EIGEN_STRONG_INLINE void _init1(const Scalar& val0, typename internal::enable_if<Base::SizeAtCompileTime==1 && internal::is_convertible<T, Scalar>::value,T>::type* = 0)
| ^~~~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: note: template argument deduction/substitution failed:
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h: In substitution of ‘template<class T> void Eigen::PlainObjectBase<Eigen::Matrix<double, 3, 1> >::_init1<T>(const Scalar&, typename Eigen::internal::enable_if<((Eigen::MatrixBase<Eigen::Matrix<double, 3, 1> >::SizeAtCompileTime == 1) && Eigen::internal::is_convertible<T, double>::value), T>::type*) [with T = std::initializer_list<double>]’:
/usr/include/eigen3/Eigen/src/Core/Matrix.h:294:31: required from ‘Eigen::Matrix<_Scalar, _Rows, _Cols, _Options, _MaxRows, _MaxCols>::Matrix(const T&) [with T = std::initializer_list<double>; _Scalar = double; int _Rows = 3; int _Cols = 1; int _Options = 0; int _MaxRows = 3; int _MaxCols = 1]’
/home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:23:81: required from ‘FramedVector3::FramedVector3(const std::initializer_list<_Tp>&) [with OtherDerived = double]’
/home/seedship/uavAP/include/uavAP/Core/SensorData.h:128:38: required from here
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: error: invalid use of incomplete type ‘struct Eigen::internal::enable_if<false, std::initializer_list<double> >’
[Truncated for brevity]
任何人都可以解释我错过了什么吗?我很困惑为什么赋值运算符起作用,但构造函数却不起作用。这发生在 Eigen 3.3.9-1 和 eigen-git.
更新:也许这个问题非常针对库,所以我在 Eigen gitlab 项目 (https://gitlab.com/libeigen/eigen/-/issues/2192)
上提问
您需要使用一对额外的括号。此外,模板化此构造函数没有意义,因为模板参数类型需要匹配特征向量的标量类型,如果您希望标量类型可配置,则需要模板化整个 class。
总结
CustomVector3(const std::initializer_list<double>& other) : Vector3({other}) {}
是执行此操作的正确方法。原因是 Eigen::Vector3d
是 Eigen::Matrix<double, 3, 1>
的类型定义。然而,矩阵的初始化列表构造函数需要处理多行和多列,因此具有签名
explicit PlainObjectBase(const std::initializer_list<std::initializer_list<Scalar>>& list);
如果您总是想使用 3D 向量,(更简单的)替代方法是使用 Eigen 的 3D 构造函数和聚合初始化:
CustomVector3(double x, double y, double z) : Vector3(x, y, z) {}
我正在尝试像这样从 Eigen 的向量 3 继承自定义 Vector3。
using Vector3 = Eigen::Vector3d;
enum class CustomEnum
{
E1, E2 E3, E4,
};
class CustomVector3 : public Vector3
{
public:
// Taken from here: https://eigen.tuxfamily.org/dox/TopicCustomizing_InheritingMatrix.html
CustomVector3():Vector3(){}
template<typename OtherDerived>
CustomVector3(const Eigen::MatrixBase<OtherDerived>& other) : Vector3(other) {}
// Initializer Constructor
template<typename OtherDerived>
CustomVector3(const std::initializer_list<OtherDerived>& other) : Vector3(other) {}
template<typename OtherDerived>
CustomVector3& operator=(const Eigen::MatrixBase<OtherDerived>& other){
this->Vector3::operator=(other);
return *this;
}
CustomEnum e = CustomEnum::E1;
};
目标是执行以下操作:
TEST_CASE("Initalizer List Test")
{
CustomVector3 vec = {1.1, 2.2, 3.3};
CHECK(vec.x() == 1.1);
CHECK(vec.y() == 2.2);
CHECK(vec.z() == 3.3);
assert(vec.frame == CustomEnum::E1);
}
没有初始化构造函数,我可以执行以下操作,但不是我上面提到的目标:
Vector3 data = {1.1, 2.2, 3.3};
CustomVector3 framedData = data;
使用初始化构造函数,我得到以下编译错误消息(我在我的代码中将其称为 FramedVector3,但为了清楚起见,我在此 post 中将其称为 CustomVector3):
/usr/include/eigen3/Eigen/src/Core/Matrix.h: In instantiation of ‘Eigen::Matrix<_Scalar, _Rows, _Cols, _Options, _MaxRows, _MaxCols>::Matrix(const T&) [with T = std::initializer_list<double>; _Scalar = double; int _Rows = 3; int _Cols = 1; int _Options = 0; int _MaxRows = 3; int _MaxCols = 1]’:
/home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:23:81: required from ‘FramedVector3::FramedVector3(const std::initializer_list<_Tp>&) [with OtherDerived = double]’
/home/seedship/uavAP/include/uavAP/Core/SensorData.h:128:38: required from here
/usr/include/eigen3/Eigen/src/Core/Matrix.h:294:31: error: no matching function for call to ‘Eigen::Matrix<double, 3, 1>::_init1<std::initializer_list<double> >(const std::initializer_list<double>&)’
294 | Base::template _init1<T>(x);
| ~~~~~~~~~~~~~~~~~~~~~~~~^~~
In file included from /usr/include/eigen3/Eigen/Core:462,
from /usr/include/eigen3/Eigen/Dense:1,
from /usr/local/include/cpsCore/Utilities/LinearAlgebra.h:13,
from /home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:9,
from /home/seedship/uavAP/include/uavAP/Core/SensorData.h:11,
from /home/seedship/uavAP/include/uavAP/Core/Orientation/ENU.h:8,
from /home/seedship/uavAP/src/Core/Orientation/ENU.cpp:5:
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:774:30: note: candidate: ‘void Eigen::PlainObjectBase<Derived>::_init1(Eigen::Index, typename Eigen::internal::enable_if<(((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime != 1) || (! Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value)) && ((! Eigen::internal::is_same<typename Eigen::internal::traits<T>::XprKind, Eigen::ArrayXpr>::value) || (typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == Eigen::Dynamic))), T>::type*) [with T = std::initializer_list<double>; Derived = Eigen::Matrix<double, 3, 1>; Eigen::Index = long int; typename Eigen::internal::enable_if<(((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime != 1) || (! Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value)) && ((! Eigen::internal::is_same<typename Eigen::internal::traits<T>::XprKind, Eigen::ArrayXpr>::value) || (typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == Eigen::Dynamic))), T>::type = std::initializer_list<double>]’
774 | EIGEN_STRONG_INLINE void _init1(Index size, typename internal::enable_if< (Base::SizeAtCompileTime!=1 || !internal::is_convertible<T, Scalar>::value)
| ^~~~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:774:43: note: no known conversion for argument 1 from ‘const std::initializer_list<double>’ to ‘Eigen::Index’ {aka ‘long int’}
774 | EIGEN_STRONG_INLINE void _init1(Index size, typename internal::enable_if< (Base::SizeAtCompileTime!=1 || !internal::is_convertible<T, Scalar>::value)
| ~~~~~~^~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: note: candidate: ‘template<class T> void Eigen::PlainObjectBase<Derived>::_init1(const Scalar&, typename Eigen::internal::enable_if<((typename Eigen::internal::dense_xpr_base<Derived>::type::SizeAtCompileTime == 1) && Eigen::internal::is_convertible<T, typename Eigen::internal::traits<T>::Scalar>::value), T>::type*) [with T = T; Derived = Eigen::Matrix<double, 3, 1>]’
788 | EIGEN_STRONG_INLINE void _init1(const Scalar& val0, typename internal::enable_if<Base::SizeAtCompileTime==1 && internal::is_convertible<T, Scalar>::value,T>::type* = 0)
| ^~~~~~
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: note: template argument deduction/substitution failed:
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h: In substitution of ‘template<class T> void Eigen::PlainObjectBase<Eigen::Matrix<double, 3, 1> >::_init1<T>(const Scalar&, typename Eigen::internal::enable_if<((Eigen::MatrixBase<Eigen::Matrix<double, 3, 1> >::SizeAtCompileTime == 1) && Eigen::internal::is_convertible<T, double>::value), T>::type*) [with T = std::initializer_list<double>]’:
/usr/include/eigen3/Eigen/src/Core/Matrix.h:294:31: required from ‘Eigen::Matrix<_Scalar, _Rows, _Cols, _Options, _MaxRows, _MaxCols>::Matrix(const T&) [with T = std::initializer_list<double>; _Scalar = double; int _Rows = 3; int _Cols = 1; int _Options = 0; int _MaxRows = 3; int _MaxCols = 1]’
/home/seedship/uavAP/include/uavAP/Core/FramedVector3.h:23:81: required from ‘FramedVector3::FramedVector3(const std::initializer_list<_Tp>&) [with OtherDerived = double]’
/home/seedship/uavAP/include/uavAP/Core/SensorData.h:128:38: required from here
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:788:30: error: invalid use of incomplete type ‘struct Eigen::internal::enable_if<false, std::initializer_list<double> >’
[Truncated for brevity]
任何人都可以解释我错过了什么吗?我很困惑为什么赋值运算符起作用,但构造函数却不起作用。这发生在 Eigen 3.3.9-1 和 eigen-git.
更新:也许这个问题非常针对库,所以我在 Eigen gitlab 项目 (https://gitlab.com/libeigen/eigen/-/issues/2192)
上提问您需要使用一对额外的括号。此外,模板化此构造函数没有意义,因为模板参数类型需要匹配特征向量的标量类型,如果您希望标量类型可配置,则需要模板化整个 class。
总结
CustomVector3(const std::initializer_list<double>& other) : Vector3({other}) {}
是执行此操作的正确方法。原因是 Eigen::Vector3d
是 Eigen::Matrix<double, 3, 1>
的类型定义。然而,矩阵的初始化列表构造函数需要处理多行和多列,因此具有签名
explicit PlainObjectBase(const std::initializer_list<std::initializer_list<Scalar>>& list);
如果您总是想使用 3D 向量,(更简单的)替代方法是使用 Eigen 的 3D 构造函数和聚合初始化:
CustomVector3(double x, double y, double z) : Vector3(x, y, z) {}