Eigen3 教程基础篇(三)

 参考

Eigen3 主页,Eigen3 官网教程

矩阵的本质,通过多种矩阵的应用去感受矩阵本质

3Blue1Brown 的线性代数,用可视化方法来表现线性代数的特性,强推

如何理解复数和虚数,有动画方便理解复数的意义

相关文章

Eigen3 教程基础篇(一)

 Eigen3 教程基础篇(二)

块操作

分块矩阵

一个分块矩阵是将矩阵分割出较小的矩形矩阵,这些较小的矩阵就称为区块。分块矩阵的分割原则是以水平线和垂直线进行划分。分块矩阵中,位在同一行(列)分区的每个子矩阵,都拥有相同的行数(列数)。

  • 分块矩阵加法:如果两个分块矩阵的每个子矩阵行列相同,那么矩阵相加等于各个子矩阵相加。

  • 分块矩阵乘法:分块矩阵 A 有 m*r 个子矩阵, 分块矩阵 B 有 r*n 个子矩阵,相乘后的矩阵 C 有 m*n 个子矩阵。

这里的 A_{i,k}*B_{k,j} 同样要满足矩阵乘法规则。

矩阵取块操作

对矩阵 matrix 的 [ i, j ] 位置,取大小 ( p, q ) 的块操作:

动态大小的块操作:

matrix.block(i,j,p,q);

固定大小的块操作:

matrix.block<p,q>(i,j);

行和列是特殊的块,可以通过 row()col()

取出来的矩阵块可以视作独立的矩阵类进行操作。

void TestBlock()
{
  Eigen::Matrix4f m;
  m << 1.2, -2.3, 3.4, -4.5,
       5.6, -6.7, 7.8, -8.9,
       9.0, -0.1, 1.2, -2.3,
       3.4, -4.5, 5.6, -6.7;
  Eigen::Matrix2f md1122 = m.block(1,1,2,2);
  Eigen::Matrix2f mf1122 = m.block<2,2>(1,1);
  ROS_INFO_STREAM("动态取块,矩阵 m 的 (1,1) 取 2*2 块:" << std::endl << md1122);
  ROS_INFO_STREAM("固定取块,矩阵 m 的 (1,1) 取 2*2 块:" << std::endl << mf1122);

  Eigen::Matrix3f n;
  n << 1.2, -2.3, 3.4,
       -4.5, 5.6, -6.7,
       7.8, -8.9, 9.0;
  Eigen::Matrix2f nd0022 = m.block(0,0,2,2);
  Eigen::Matrix2f mn22 = md1122 + nd0022;
  ROS_INFO_STREAM("动态取块,矩阵 n 的 (0,0) 取 2*2 块:" << std::endl << nd0022);
  ROS_INFO_STREAM("矩阵 m(1,1){2*2} + n(0,0){2*2}" << std::endl << mn22);

  Eigen::Matrix2f md2022 = m.block(2,0,2,2);
  Eigen::Matrix2f nf1122 = n.block<2,2>(1,1);
  ROS_INFO_STREAM("矩阵 m(2,0){2*2} * n(1,1){2*2}" << std::endl << md2022*nf1122);
}

向量取块操作

取 n 个向量头部元素的块,动态块操作:vector.head(n),固定区块:vector.head<n>()

取 n 个向量尾部元素的块,动态块操作:vector.tail(n),固定区块:vector.tail<n>()

在 i 位置取 n 个元素的块,动态块操作:vector.segment(i,n),固定区块:vector.segment<n>(i)

void TestVectorBlock()
{
  Eigen::Vector3f v3f(1.2, -2.3, 3.4);
  Eigen::Vector2f v3fh2 = v3f.head(2);
  Eigen::Vector2f v3ft2 = v3f.tail(2);
  ROS_INFO_STREAM("向量 v3f:" << std::endl << v3f);
  ROS_INFO_STREAM("向量 v3f 取头部 2 个元素:" << std::endl << v3fh2);
  ROS_INFO_STREAM("向量 v3f 取尾部 2 个元素:" << std::endl << v3ft2);

  Eigen::VectorXf v6f(6);
  v6f << 0.1, -1.2, 2.3, -3.4, 4.5, -5.6;
  Eigen::Vector3f v6f33 = v6f.segment(3,3);
  Eigen::Vector3f v6f23 = v6f.segment<3>(2);
  ROS_INFO_STREAM("向量 v6f:" << std::endl << v6f);
  ROS_INFO_STREAM("向量 v6f 动态取块在 3 位置取 3 个元素:" << std::endl << v6f33);
  ROS_INFO_STREAM("向量 v6f 在 2 位置固定取块 3 个元素:" << std::endl << v6f23);
}

Reductions 归约

在 Eigen 中,Reductions(归约)是返回单个标量值的函数方法。

  • 常见的归约方法:

sum() 矩阵系数和,prod() 矩阵系数乘积,mean() 矩阵系数均值,minCoeff() 矩阵中最小系数,maxCoeff() 矩阵中最大系数,trace() 矩阵中对角线(左上角到右下角)元素之和。

squaredNorm() 向量系数绝对值的平方之和,norm() 向量系数绝对值平方和的平方根。

这两个方法对矩阵同样适用,此时矩阵 M(n,p) 被看做 n*p 大小的向量。

  • 布尔归约:

all(),如果矩阵,数组的所有系数均计算为 true,返回 true;

any(),如果矩阵,数组的存在任一系数计算为 true,返回 true;

count(),返回矩阵或数组中系数计算为 true 的数量;

  • 部分归约:

colwise() 和 rowwise() 可以取得列向,或行向的归约。

void TestReductions()
{
  Eigen::Matrix4f m;
  m << 1.2, -2.3, 3.4, -4.5,
       5.6, -6.7, 7.8, -8.9,
       9.0, -0.1, 1.2, -2.3,
       3.4, -4.5, 5.6, -6.7;
  ROS_INFO_STREAM("矩阵 m:" << std::endl << m);

  ROS_INFO_STREAM("m.sum():" << m.sum() << std::endl <<
                  "m.prod():" << m.prod() << std::endl <<
                  "m.mean():" << m.mean() << std::endl <<
                  "m.minCoeff():" << m.minCoeff() << std::endl <<
                  "m.maxCoeff():" << m.maxCoeff() << std::endl <<
                  "m.trace():" << m.trace() << std::endl <<
                  "m.squaredNorm():" << m.squaredNorm() << std::endl <<
                  "m.norm():" << m.norm() << std::endl);

  ROS_INFO_STREAM("m.colwise().maxCoeff():" << m.colwise().maxCoeff() << std::endl <<
                  "m.rowwise().sum().minCoeff():" << m.rowwise().sum().minCoeff() << std::endl);
}

broadcasting 传播

对所有的方向进行操作。

void TestBroadcasting()
{
  Eigen::MatrixXf m(2,4);
  m << 1.0, -11.23, 6.9, -0.83,
       -5.6, 2.56, -2.97, 6.56;
  ROS_INFO_STREAM("矩阵 m:" << std::endl << m);

  Eigen::VectorXf n(2);
  n << 2,
       3;
  ROS_INFO_STREAM("向量 n:" << std::endl << n);

  Eigen::Index index;
  (m.colwise() - n).colwise().squaredNorm().minCoeff(&index);
  std::cout << "Nearest neighbour is column " << index << ":" << std::endl;
  std::cout << m.col(index) << std::endl;
}

Map 类:原始缓存数据接口

当在 Eigen 之外定义了一组数据,这组数据就是一个矩阵或者向量。这里可以通过 Eigen 的 Map 模板来使用 Eigen 的算法处理这组数据。

Map 模板类:

Map<Matrix<typename Scalar, int RowsAtCompileTime, int ColsAtCompileTime> >

例如Map<MatrixXf> mf(pf,rows,cols),pf 是 Eigen 之外定义的 rows*cols 大小的矩阵的数据内存地址,这里通过 Map 模板构造的对象 mf 得到了和 MatrixXf 等效的数据处理接口,可以像使用 MatrixXf 一样处理 pf 地址的数据。

修改 Map 模板类对象的数据地址,挺奇怪的写法:

new (&map) Map<Matrix<typename Scalar, int Rows, int Cols> >(new_data);
void TestMap()
{
  int array[8];
  for(int i = 0; i < 8; ++i)
    array[i] = i;
  auto m = Eigen::Map<Eigen::Matrix<int,2,4>>(array);
  ROS_INFO_STREAM("矩阵 array -> m:" << std::endl << m);
  ROS_INFO_STREAM("矩阵 array -> m.colwise().sum()" << std::endl << m.colwise().sum());

  int data[] = {1,2,3,4,5,6,7,8,9};
  new (&m) Eigen::Map<Eigen::Matrix<int,2,4>>(data);
  ROS_INFO_STREAM("矩阵 data -> m:" << std::endl << m);
  ROS_INFO_STREAM("矩阵 data -> m.rowwise().squaredNorm():" << std::endl << m.rowwise().squaredNorm());
}

Aliasing 混叠

在 Eigen 中,Aliasing 混叠指赋值语句左右两侧出现相同的矩阵/数组/向量,这里指左右操作数内存上有重叠。有时候 Aliasing 混叠会导致异常。

  • 混叠异常情况

void TestAliasing()
{
  Eigen::MatrixXi m(3,3); 
  m << 1, 2, 3,
       4, 5, 6,
       7, 8, 9;
  ROS_INFO_STREAM("矩阵 m:" << std::endl << m);
  
  m.bottomRightCorner(2,2) = m.topLeftCorner(2,2);
  ROS_INFO_STREAM("after the assignment, m:" << std::endl << m);
}

上面的例子中,系数 5 同时存在于左上 2*2 矩阵块和右下 2*2 矩阵块中,Eigen 一般情况下为了性能直接修改数据内存,而没有拷贝右值的临时变量处理。这就导致了混叠操作有可能出现计算错误。

其他像 a = a.transpose()的操作也有导致出错!

  • 避免混叠异常

通过 eval() 函数给右值创建临时对象。

void TestAliasing()
{
  ROS_INFO_STREAM("---------- 混叠异常 ----------");
  Eigen::MatrixXi m(3,3); 
  m << 1, 2, 3,
       4, 5, 6,
       7, 8, 9;
  auto x = m;
  ROS_INFO_STREAM("矩阵 m:" << std::endl << m);
  
  m.bottomRightCorner(2,2) = m.topLeftCorner(2,2);
  ROS_INFO_STREAM("after the assignment, m:" << std::endl << m);

  ROS_INFO_STREAM("---------- 避免混叠 ----------");
  x.bottomRightCorner(2,2) = x.topLeftCorner(2,2).eval();
  ROS_INFO_STREAM("after the assignment, x:" << std::endl << x);
}

另外还有 ***InPlace() 接口避免混叠异常:

矩阵乘法是唯一被默认假定混叠的运算,所以矩阵乘法是默认通过临时对象计算,此时混叠不会出现异常。

也是由于矩阵乘法的特殊处理,其中的临时对象赋值和销毁消耗了时间,此时可以通过 noalias() 函数指定矩阵乘法不使用临时对象。

下面用 noalias() 告诉 eigen 库,左值的矩阵数据内存不与右值操作数有重叠,不需要拷贝右值的临时变量。

void TestAliasingMultiplication()
{
  Eigen::MatrixXi m(3,3); 
  m << 1, 2, 3,
       4, 5, 6,
       7, 8, 9;
  auto x = m;
  ROS_INFO_STREAM("矩阵 m:" << std::endl << m);

  auto mx = m * x;
  ROS_INFO_STREAM("矩阵 m * 矩阵 x:" << std::endl << mx);

  m *= m;
  ROS_INFO_STREAM("矩阵 m 乘法混叠无异常, m:" << std::endl << m);

  x.noalias() = x * x;
  ROS_INFO_STREAM("矩阵 x 混叠,不使用临时变量, x:" << std::endl << x);
}

一般来说,可以用 noalias() 指定不创建临时对象,加速计算;用 eval() 来创建临时对象,避免计算错误。

Explanation of the assertion on unaligned arrays 未对齐内存问题

使用 eigen 库时候,可能出现下面的断言错误:

这个错误的根本原因是固定矩阵/向量大小的 Eigen 对象没有在正确对齐的位置创建,导致 SIMD 指令寻址错误。

!!!使用 C++17 可以避免这个问题!!!

!!!动态大小的矩阵不会有这个问题,Eigen 会管理动态内存的对象!!!

常见的出错场景(C++11编译测试)

  • 构建具有固定大小的 Eigen 成员变量的对象

class xclass
{
 public:
  Eigen::Matrix2d v;
};

void TestUnalignedArray()
{
  auto x = new xclass();
  x->v << 1.0, 2.0, 3.0, 4.0;
  ROS_INFO_STREAM("x.v:" << std::endl << x->v);
}

未出现上面的错误:

  • 使用容器存储 Eigen 对象

void TestUnalignedArray()
{
  std::vector<Eigen::Vector2d> ve2d;
  Eigen::Vector2d m2d0;
  m2d0 << 1.0, 2.0;
  Eigen::Vector2d m2d1;
  m2d1 << 5.0, 6.0;

  ve2d.push_back(m2d0);
  ve2d.push_back(m2d1);
  ROS_INFO_STREAM("ve2d.front()" << std::endl << ve2d.front());
  ROS_INFO_STREAM("ve2d.at(1)" << std::endl << ve2d.at(1));
}

未出现上面的错误:

  • 按值传递 Eigen 对象

void PrintInputMatrix(const Eigen::Matrix2d m)
{
  ROS_INFO_STREAM("input m:" << std::endl << m);
}

void TestUnalignedArray()
{
  Eigen::Matrix2d m;
  m << 1.2, 2.3, 3.4, 4.5;
  PrintInputMatrix(m);
}

未出现上面的错误:

  • 在堆栈创建 Eigen 对象(Win 中的 gcc)

void foo()
{
  Eigen::Quaternionf q;
  //...
}

这个就不测试了...

不知道为啥就是没出现异常 :) 具体的解决办法参考原文资料吧。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值