本篇内容主要解决Eigen大型矩阵的构建、赋值、运算过程中,内存溢出的问题,报错:OBJECT_ALLOCATED_ON_STACK_IS_TOO_BIG!
在使用Eigen中,创建高维矩阵和运算的时候,一直报一个错误,大致内容如下:
In file included from /usr/include/eigen3/Eigen/Core:297:0,
from /home/mm/Cpptest/untitled3/main.cpp:3:
/usr/include/eigen3/Eigen/src/Core/DenseStorage.h: In instantiation of ‘void Eigen::internal::check_static_allocation_size() [with T = double; int Size = 1562500]’:
/usr/include/eigen3/Eigen/src/Core/DenseStorage.h:110:41: required from ‘Eigen::internal::plain_array<T, Size, MatrixOrArrayOptions, 16>::plain_array() [with T = double; int Size = 1562500; int MatrixOrArrayOptions = 0]’
/usr/include/eigen3/Eigen/src/Core/DenseStorage.h:187:38: required from ‘Eigen::DenseStorage<T, Size, _Rows, _Cols, _Options>::DenseStorage() [with T = double; int Size = 1562500; int _Rows = 1250; int _Cols = 1250; int _Options = 0]’
/usr/include/eigen3/Eigen/src/Core/PlainObjectBase.h:457:55: required from ‘Eigen::PlainObjectBase<Derived>::PlainObjectBase() [with Derived = Eigen::Matrix<double, 1250, 1250>]’
/usr/include/eigen3/Eigen/src/Core/Matrix.h:259:41: required from ‘Eigen::Matrix<_Scalar, _Rows, _Cols, _Options, _MaxRows, _MaxCols>::Matrix() [with _Scalar = double; int _Rows = 1250; int _Cols = 1250; int _Options = 0; int _MaxRows = 1250; int _MaxCols = 1250]’
/home/mm/Cpptest/untitled3/main.cpp:11:53: required from here
/usr/include/eigen3/Eigen/src/Core/util/StaticAssert.h:32:40: error: static assertion failed: OBJECT_ALLOCATED_ON_STACK_IS_TOO_BIG
#define EIGEN_STATIC_ASSERT(X,MSG) static_assert(X,#MSG);
^
/usr/include/eigen3/Eigen/src/Core/DenseStorage.h:33:3: note: in expansion of macro ‘EIGEN_STATIC_ASSERT’
EIGEN_STATIC_ASSERT(Size * sizeof(T) <= EIGEN_STACK_ALLOCATION_LIMIT, OBJECT_ALLOCATED_ON_STACK_IS_TOO_BIG);
这里面最重要的一个信息就是,下面这一个错误提示:
error: static assertion failed: OBJECT_ALLOCATED_ON_STACK_IS_TOO_BIG
#define EIGEN_STATIC_ASSERT(X,MSG) static_assert(X,#MSG);
主要原因就是:栈溢出
要想解决这个问题,我们只需要将创建的静态矩阵,改成动态矩阵。
例如,先前我们创建的如下面形式的矩阵:
Eigen::Matrix matrix_NN;
将上一行代码改成下面形式:
Eigen::Matrix matrix_NN;
这里的Eigen::Dynamic
表示的是矩阵的维数是动态的,在赋值或者运算的时候,自动产生对应的维数大小。
注意:因为我们定义的矩阵是动态维数,所以有时候可能必须要先“固定(有别于静态矩阵的固定)”维数,那么可以通过赋值的方式,下面介绍几个赋值方式使其产生“固定”维数的方法:
1).用随机值矩阵初始化
matrix_NN = Eigen::MatrixXd::Random(500, 500);
2).用单位矩阵初始化
matrix_NN = Eigen::MatrixXd::Identity(500, 500);
3).用零矩阵初始化
matrix_NN = Eigen::MatrixXd::Zero(500, 500);
有一点需要强调: 虽然我们可以使用任何维度的矩阵去赋值或者初始化matrix_NN
,但是始终没有改变它是动态矩阵的性质,也就是维数运行过程始终可以变化,这一点儿不同于静态矩阵,静态矩阵编译时就确定了维度,运行时不能变。
补充:静态矩阵和动态矩阵的区别:
动态矩阵和静态矩阵:动态矩阵是指其大小在运行时确定,静态矩阵是指其大小在编译时确定。
MatrixXd
:表示任意大小的元素类型为double的矩阵变量,其大小只有在运行时被赋值之后才能知道。
Matrix3d
:表示元素类型为double大小为3*3的矩阵变量,其大小在编译时就知道。
矩阵使用了动态矩阵之后,就可以做大型矩阵的运算了,我将一个小例子放在这里与大家分享,我将一个400维的矩阵,进行方程求解,在我的已经用了好多年的美帝良心机(联想)笔记本上,分别使用直接求逆的方法和QR分解的方法,运行时间如下:
Direct guidance uses times: 315363ms
QR uses times: 832.552ms
代码如下,如果你有兴趣可以在你的电脑上试一下:
#include
#include
#include
#include
using namespace std;
#define MATRIX_SIZE 400
int main() {
//This codes mainly use to solve the equation, matrix_NN *x = v_Nd
Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic> matrix_NN;
matrix_NN = Eigen::MatrixXd::Random(MATRIX_SIZE, MATRIX_SIZE);
Eigen::Matrix<double, MATRIX_SIZE, 1> v_Nd;
v_Nd = Eigen::MatrixXd::Random(MATRIX_SIZE, 1);
clock_t time_stt = clock();
Eigen::Matrix<double, MATRIX_SIZE, 1> x = matrix_NN.inverse() * v_Nd;
cout << "Direct guidance uses times: " << 1000 * (clock() - time_stt) / (double)CLOCKS_PER_SEC << "ms" << endl;
time_stt = clock();
x = matrix_NN.colPivHouseholderQr().solve(v_Nd);
cout << "QR uses times: " << 1000 * (clock() - time_stt) / (double)CLOCKS_PER_SEC << "ms" << endl;
return 0;
}