Eigen库-eigen代数运算(逆,特征值等)

1 Eigen数据类型

Eigen是一个模板类,基本数据单元为矩阵,它的前3个参数为:数据类型,行,列,如下所示:
Eigen::Matrix<float,2,3> matrix_23;

Eigen通过tyepdef定义了许多内置类型,不过底层仍然是Eigen::Matrix,如下所示:
Eigen::Matrix3d matrix_33 = Eigen::Matrix3d::Zero():     //初始化为0       //Matrix3d实质上是Eigen::Matrix<double, 3, 3>
Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic> matrix_dynamic;      //如果不确定矩阵大小,可以使用动态大小的矩阵
Eigen::MatrixXd matrix_xd;                                                 //与上类似,表示任意大小的元素类型为double的矩阵变量
Eigen::MatrixXf matrix_xf; 
Eigen::Vector3d v_3d;                                                      //Vector3d实质上是Eigen::Matrix<double, 3, 1>

2 程序实例

#include <iostream>
#include <Eigen/core>   //eigen部分
#include <Eigen/Dense>  //稠密矩阵的代数运算
#include <ctime>
//#include <vector>
using namespace std;
using namespace Eigen; 

#define MATRIX_SIZE 50

int main(int argc, char *argv[])
{
	Matrix2d a;
	a << 1, 2,
		 3, 4;
	MatrixXd b(2, 2);
	b << 2, 2,
		 2, 2;
	cout << "a + b = \n" << a + b << endl<<endl;
	cout << "Doing a += b;" << endl;
	a += b;
	cout << "Now a = \n" << a <<endl<< endl;

	cout << "a^T = \n " << a.transpose() << endl<<endl; 
	cout << "a*b = \n " << a*b <<endl<< endl;

	Vector3d v(1, 2, 3);
	Vector3d w(1, 0, 0);
	cout << v <<endl<< endl;
	cout << v.transpose() <<endl<< endl;
	cout << "v-w =\n" << v-w  << endl<<endl;
	

	Eigen::Matrix3d matrix_33 = Eigen::Matrix3d::Random();
	cout << matrix_33 << endl << endl;

	cout << matrix_33.transpose() << endl;    //转置
	cout << matrix_33.sum() << endl;          //各元素和
	cout << matrix_33.trace() << endl;         //迹
	cout << matrix_33 * 10 << endl;            //数乘
	cout << matrix_33.inverse() << endl;        //逆
	cout << matrix_33.determinant() << endl;     //行列式

	// 求特征值及特征向量
	// 实对称矩阵可以保证对角化成功
	Eigen::SelfAdjointEigenSolver<Eigen::Matrix3d> eigen_solver(matrix_33.transpose()*matrix_33);
	cout << "特征值eigenvalues = \n" << eigen_solver.eigenvalues() << endl;
	cout << "特征向量eigenvectors = \n" << eigen_solver.eigenvectors() << endl;

	
	// 求解 matrix_NN * x = v_Nd 这个方程
	// N的大小在前边的宏里定义,它由随机数生成
	// 直接求逆自然是最直接的,但是求逆运算量大
	Eigen::Matrix< double, 50, 50 > matrix_NN = Eigen::MatrixXd::Random(50, 50);   //初始化为随机数
	Eigen::Matrix< double, 50, 1> v_Nd = Eigen::MatrixXd::Random(50, 1);

    // 解方程
	// 直接求逆
	clock_t time_stt = clock(); // 计时 ,头文件#include <ctime>
	Eigen::Matrix<double, MATRIX_SIZE, 1> x = matrix_NN.inverse()*v_Nd;
	cout << "time use in normal inverse is " << 1000 * (clock() - time_stt) / (double)CLOCKS_PER_SEC << "ms" << endl;

	// 通常用矩阵分解来求,例如QR分解,速度会快很多
	time_stt = clock();
	x = matrix_NN.colPivHouseholderQr().solve(v_Nd);
	cout << "time use in Qr decomposition is " << 1000 * (clock() - time_stt) / (double)CLOCKS_PER_SEC << "ms" << endl;

	system("PAUSE");
	return 0;
}

运行结果:

a + b =
3 4
5 6

Doing a += b;
Now a =
3 4
5 6

a^T =
 3 5
4 6

a*b =
 14 14
22 22

1
2
3

1 2 3

v-w =
0
2
3

 -0.997497   0.617481  -0.299417
  0.127171   0.170019   0.791925
 -0.613392 -0.0402539    0.64568

 -0.997497   0.127171  -0.613392
  0.617481   0.170019 -0.0402539
 -0.299417   0.791925    0.64568
0.401715
-0.181799
 -9.97497   6.17481  -2.99417
  1.27171   1.70019   7.91925
 -6.13392 -0.402539    6.4568
-0.271556    0.7412  -1.03501
  1.08862   1.58676  -1.44134
-0.190108  0.803059  0.475647
-0.521644
特征值eigenvalues =
0.145004
   1.136
 1.65193
特征向量eigenvectors =
 0.415633 0.0900561 -0.905063
 0.906947 0.0339438  0.419875
0.0685336 -0.995358 -0.067568
time use in normal inverse is 16ms
time use in Qr decomposition is 0ms
请按任意键继续. . .
  • 1
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
不使用Eigen特征值可以利用迭代法或者直接解特征值方程的方法。以下是两种方法的简单介绍: 1. 迭代法 迭代法是一种通过迭代逼近特征值的方法,常用的方法有幂法和反幂法。幂法是通过多次矩阵与向量的乘积来逼近矩阵的最大特征值和对应的特征向量,而反幂法是通过解矩阵的来逼近矩阵的最小特征值和对应的特征向量。 2. 直接解特征值方程 特征值方程是一个关于特征值的多项式方程,直接解该方程可以得到矩阵的所有特征值。对于一个n阶矩阵,其特征值方程为:det(A - λI) = 0,其中I是n阶单位矩阵,det表示行列式。该方程的解为矩阵的n个特征值。 以下是使用迭代法解矩阵特征值的示例代码: ```c++ #include <iostream> #include <vector> #include <cmath> using namespace std; const double EPS = 1e-8; // 精度 const int MAX_ITER = 1000; // 最大迭代次数 // 幂法解最大特征值和对应的特征向量 void power_method(vector<vector<double>>& A, double& lambda, vector<double>& x) { int n = A.size(); double norm = 0.0; int iter = 0; x.assign(n, 1.0); // 初始向量 while (iter < MAX_ITER) { // 计算新向量 vector<double> y(n); for (int i = 0; i < n; i++) { double sum = 0.0; for (int j = 0; j < n; j++) sum += A[i][j] * x[j]; y[i] = sum; } // 计算新向量的模长 double new_norm = 0.0; for (int i = 0; i < n; i++) new_norm += y[i] * y[i]; new_norm = sqrt(new_norm); // 归一化新向量 for (int i = 0; i < n; i++) y[i] /= new_norm; // 计算特征值的变化量 double delta = 0.0; for (int i = 0; i < n; i++) delta += (y[i] - x[i]) * (y[i] - x[i]); delta = sqrt(delta); // 更新特征值特征向量 lambda = new_norm; x = y; // 判断是否满足精度要 if (delta < EPS) break; iter++; } } int main() { vector<vector<double>> A = {{1.0, 2.0, 3.0}, {4.0, 5.0, 6.0}, {7.0, 8.0, 9.0}}; double lambda; vector<double> x; power_method(A, lambda, x); cout << "The maximum eigenvalue is " << lambda << endl; cout << "The corresponding eigenvector is "; for (int i = 0; i < x.size(); i++) cout << x[i] << " "; cout << endl; return 0; } ``` 该代码使用幂法解矩阵的最大特征值和对应的特征向量。首先定义了一个3x3的矩阵A,然后使用`power_method`函数进行幂法迭代,得到最大特征值和对应的特征向量

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值