Eigen介绍及简单使用
博客转载自:https://blog.csdn.net/fengbingchun/article/details/47378515
Eigen是可以用来进行线性代数、矩阵、向量操作等运算的C++库,它里面包含了很多算法。它的License是MPL2。它支持多平台。Eigen采用源码的方式提供给用户使用,在使用时只需要包含Eigen的头文件即可进行使用。之所以采用这种方式,是因为Eigen采用模板方式实现,由于模板函数不支持分离编译,所以只能提供源码而不是动态库的方式供用户使用。 矩阵的定义:Eigen中关于矩阵类的模板函数中,共有六个模板参数,常用的只有前三个。其前三个参数分别表示矩阵元素的类型、行数和列数。 矩阵定义时可以使用Dynamic来表示矩阵的行列数为未知。Eigen中无论是矩阵还是数组、向量,无论是静态矩阵还是动态矩阵都提供默认构造函数,也就是定义这些数据结构时都可以不用提供任何参数,其大小均由运行时来确定。矩阵的构造函数中只提供行列数、元素类型的构造参数,而不提供元素值的构造,对于比较小的、固定长度的向量提供初始化元素的定义。
矩阵类型:Eigen中的矩阵类型一般都是用类似MatrixXXX来表示,可以根据该名字来判断其数据类型,比如”d”表示double类型,”f”表示float类型,”i”表示整数,”c”表示复数;Matrix2f,表示的是一个2*2维的,其每个元素都是float类型
数据存储:Matrix创建的矩阵默认是按列存储,Eigen在处理按列存储的矩阵时会更加高效。如果想修改可以在创建矩阵的时候加入参数,如:
- Matrix<int,3, 4, ColMajor> Acolmajor;
- Matrix<int,3, 4, RowMajor> Arowmajor;
动态矩阵和静态矩阵:动态矩阵是指其大小在运行时确定,静态矩阵是指其大小在编译时确定。MatrixXd:表示任意大小的元素类型为double的矩阵变量,其大小只有在运行时被赋值之后才能知道。Matrix3d:表示元素类型为double大小为3*3的矩阵变量,其大小在编译时就知道。 在Eigen中行优先的矩阵会在其名字中包含有row,否则就是列优先。 Eigen中的向量只是一个特殊的矩阵,其维度为1而已。
矩阵元素的访问:在矩阵的访问中,行索引总是作为第一个参数,Eigen中矩阵、数组、向量的下标都是从0开始。矩阵元素的访问可以通过”()”操作符完成。例如m(2, 3)既是获取矩阵m的第2行第3列元素。 针对向量还提供”[]”操作符,注意矩阵则不可如此使用。
设置矩阵的元素:在Eigen中重载了”<<”操作符,通过该操作符即可以一个一个元素的进行赋值,也可以一块一块的赋值。另外也可以使用下标进行赋值。
重置矩阵大小:当前矩阵的行数、列数、大小可以通过rows()、cols()和size()来获取,对于动态矩阵可以通过resize()函数来动态修改矩阵的大小。注意:(1)、固定大小的矩阵是不能使用resize()来修改矩阵的大小;(2)、resize()函数会析构掉原来的数据,因此调用resize()函数之后将不能保证元素的值不改变;(3)、使用”=”操作符操作动态矩阵时,如果左右两边的矩阵大小不等,则左边的动态矩阵的大小会被修改为右边的大小。
如何选择动态矩阵和静态矩阵:对于小矩阵(一般小于16)使用固定大小的静态矩阵,它可以带来比较高的效率;对于大矩阵(一般大于32)建议使用动态矩阵。注意:如果特别大的矩阵使用了固定大小的静态矩阵则可能会造成栈溢出的问题。
矩阵和向量的算术运算:在Eigen中算术运算重载了C++的+、-、*
(1)、矩阵的运算:提供+、-、一元操作符”-”、+=、-=;二元操作符+/-,表示两矩阵相加(矩阵中对应元素相加/减,返回一个临时矩阵);一元操作符-表示对矩阵取负(矩阵中对应元素取负,返回一个临时矩阵);组合操作法+=或者-=表示(对应每个元素都做相应操作);矩阵还提供与标量(单一数字)的乘除操作,表示每个元素都与该标量进行乘除操作;
(2)、求矩阵的转置、共轭矩阵、伴随矩阵:可以通过成员函数transpose()、conjugate()、adjoint()来完成。注意:这些函数返回操作后的结果,而不会对原矩阵的元素进行直接操作,如果要让原矩阵进行转换,则需要使用响应的InPlace函数,如transpoceInPlace()等;
(3)、矩阵相乘、矩阵向量相乘:使用操作符*,共有*和*=两种操作符;
(4)、矩阵的块操作:有两种使用方法
- matrix.block(i,j, p, q) : 表示返回从矩阵(i, j)开始,每行取p个元素,每列取q个元素所组成的临时新矩阵对象,原矩阵的元素不变;
- matrix.block<p,q>(i, j) :<p, q>可理解为一个p行q列的子矩阵,该定义表示从原矩阵中第(i, j)开始,获取一个p行q列的子矩阵,返回该子矩阵组成的临时矩阵对象,原矩阵的元素不变;
(5)、向量的块操作
- 获取向量的前n个元素:vector.head(n);
- 获取向量尾部的n个元素:vector.tail(n);
- 获取从向量的第i个元素开始的n个元素:vector.segment(i,n);
- Map类:在已经存在的矩阵或向量中,不必拷贝对象,而是直接在该对象的内存上进行运算操作。
Eigen例子:https://github.com/fengbingchun/Eigen_Test
- #include "stdafx.h"
- #include <iostream>
- #include <Eigen/Dense>
- template <typename T>
- static void matrix_mul_matrix(T* p1, int iRow1, int iCol1, T* p2, int iRow2, int iCol2, T* p3)
- {
- if (iRow1 != iRow2) return;
- //列优先
- //Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic> > map1(p1, iRow1, iCol1);
- //Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic> > map2(p2, iRow2, iCol2);
- //Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic> > map3(p3, iCol1, iCol2);
- //行优先
- Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor> > map1(p1, iRow1, iCol1);
- Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor> > map2(p2, iRow2, iCol2);
- Eigen::Map< Eigen::Matrix<T, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor> > map3(p3, iCol1, iCol2);
- map3 = map1 * map2;
- }
- int main(int argc, char* argv[])
- {
- //1. 矩阵的定义
- Eigen::MatrixXd m(2, 2);
- Eigen::Vector3d vec3d;
- Eigen::Vector4d vec4d(1.0, 2.0, 3.0, 4.0);
- //2. 动态矩阵、静态矩阵
- Eigen::MatrixXd matrixXd;
- Eigen::Matrix3d matrix3d;
- //3. 矩阵元素的访问
- m(0, 0) = 1;
- m(0, 1) = 2;
- m(1, 0) = m(0, 0) + 3;
- m(1, 1) = m(0, 0) * m(0, 1);
- std::cout << m << std::endl << std::endl;
- //4. 设置矩阵的元素
- m << -1.5, 2.4,
- 6.7, 2.0;
- std::cout << m << std::endl << std::endl;
- int row = 4;
- int col = 5;
- Eigen::MatrixXf matrixXf(row, col);
- matrixXf << 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20;
- std::cout << matrixXf << std::endl << std::endl;
- matrixXf << Eigen::MatrixXf::Identity(row, col);
- std::cout << matrixXf << std::endl << std::endl;
- //5. 重置矩阵大小
- Eigen::MatrixXd matrixXd1(3, 3);
- m = matrixXd1;
- std::cout << m.rows() << " " << m.cols() << std::endl << std::endl;
- //6. 矩阵运算
- m << 1, 2, 7,
- 3, 4, 8,
- 5, 6, 9;
- std::cout << m << std::endl;
- matrixXd1 = Eigen::Matrix3d::Random();
- m += matrixXd1;
- std::cout << m << std::endl << std::endl;
- m *= 2;
- std::cout << m << std::endl << std::endl;
- std::cout << -m << std::endl << std::endl;
- std::cout << m << std::endl << std::endl;
- //7. 求矩阵的转置、共轭矩阵、伴随矩阵
- std::cout << m.transpose() << std::endl << std::endl;
- std::cout << m.conjugate() << std::endl << std::endl;
- std::cout << m.adjoint() << std::endl << std::endl;
- std::cout << m << std::endl << std::endl;
- m.transposeInPlace();
- std::cout << m << std::endl << std::endl;
- //8. 矩阵相乘、矩阵向量相乘
- std::cout << m*m << std::endl << std::endl;
- vec3d = Eigen::Vector3d(1, 2, 3);
- std::cout << m * vec3d << std::endl << std::endl;
- std::cout << vec3d.transpose()*m << std::endl << std::endl;
- //9. 矩阵的块操作
- std::cout << m << std::endl << std::endl;
- std::cout << m.block(1, 1, 2, 2) << std::endl << std::endl;
- std::cout << m.block<1, 2>(0, 0) << std::endl << std::endl;
- std::cout << m.col(1) << std::endl << std::endl;
- std::cout << m.row(0) << std::endl << std::endl;
- //10. 向量的块操作
- Eigen::ArrayXf arrayXf(10);
- arrayXf << 1, 2, 3, 4, 5, 6, 7, 8, 9, 10;
- std::cout << vec3d << std::endl << std::endl;
- std::cout << arrayXf << std::endl << std::endl;
- std::cout << arrayXf.head(5) << std::endl << std::endl;
- std::cout << arrayXf.tail(4) * 2 << std::endl << std::endl;
- //11. 求解矩阵的特征值和特征向量
- Eigen::Matrix2f matrix2f;
- matrix2f << 1, 2, 3, 4;
- Eigen::SelfAdjointEigenSolver<Eigen::Matrix2f> eigenSolver(matrix2f);
- if (eigenSolver.info() == Eigen::Success) {
- std::cout << eigenSolver.eigenvalues() << std::endl << std::endl;
- std::cout << eigenSolver.eigenvectors() << std::endl << std::endl;
- }
- //12. 类Map及动态矩阵的使用
- int array1[4] = { 1, 2, 3, 4 };
- int array2[4] = { 5, 6, 7, 8 };
- int array3[4] = { 0, 0, 0, 0};
- matrix_mul_matrix(array1, 2, 2, array2, 2, 2, array3);
- for (int i = 0; i < 4; i++)
- std::cout << array3[i] << std::endl;
- return 0;
- }
Eigen介绍及简单使用的更多相关文章
- 【转载】Ssh整合开发介绍和简单的登入案例实现
Ssh整合开发介绍和简单的登入案例实现 Ssh整合开发介绍和简单的登入案例实现 一 介绍: Ssh是strtus2-2.3.1.2+ spring-2.5.6+hibernate-3.6.8整合的开 ...
- python模块介绍- HTMLParser 简单的HTML和XHTML解析器
python模块介绍- HTMLParser 简单的HTML和XHTML解析器 2013-09-11 磁针石 #承接软件自动化实施与培训等gtalk:ouyangchongwu#gmail.comqq ...
- 基于.NET CORE微服务框架 -surging的介绍和简单示例 (开源)
一.前言 至今为止编程开发已经11个年头,从 VB6.0,ASP时代到ASP.NET再到MVC, 从中见证了.NET技术发展,从无畏无知的懵懂少年,到现在的中年大叔,从中的酸甜苦辣也只有本人自知.随着 ...
- WebRTC介绍及简单应用
WebRTC介绍及简单应用 WebRTC,即Web Real-Time Communication,web实时通信技术.简单地说就是在web浏览器里面引入实时通信,包括音视频通话等. WebRTC实时 ...
- 1. pyhanlp介绍和简单应用
1. pyhanlp介绍和简单应用 2. 观点提取和聚类代码详解 1. 前言 中文分词≠自然语言处理! 中文分词只是第一步:HanLP从中文分词开始,覆盖词性标注.命名实体识别.句法分析.文本分类等常 ...
- C#串口介绍以及简单串口通信程序设计实现
C#串口介绍以及简单串口通信程序设计实现 周末,没事干,写个简单的串口通信工具,也算是本周末曾来过,废话不多,直接到主题 串口介绍 串行接口简称串口,也称串行通信接口或串行通讯接口(通常指COM接口) ...
- 消息队列介绍、RabbitMQ&Redis的重点介绍与简单应用
消息队列介绍.RabbitMQ&Redis的重点介绍与简单应用 消息队列介绍.RabbitMQ.Redis 一.什么是消息队列 这个概念我们百度Google能查到一大堆文章,所以我就通俗的讲下 ...
- 进击的Python【第十二章】:mysql介绍与简单操作,sqlachemy介绍与简单应用
进击的Python[第十二章]:mysql介绍与简单操作,sqlachemy介绍与简单应用 一.数据库介绍 什么是数据库? 数据库(Database)是按照数据结构来组织.存储和管理数据的仓库,每个数 ...
- 进击的Python【第十一章】:消息队列介绍、RabbitMQ&Redis的重点介绍与简单应用
消息队列介绍.RabbitMQ.Redis 一.什么是消息队列 这个概念我们百度Google能查到一大堆文章,所以我就通俗的讲下消息队列的基本思路. 还记得原来写过Queue的文章,不管是线程queu ...
随机推荐
- (转) Android Fragment完全解析,关于碎片你所需知道的一切
我们都知道,Android上的界面展示都是通过Activity实现的,Activity实在是太常用了,我相信大家都已经非常熟悉了,这里就不再赘述. 但是Activity也有它的局限性,同样的界面在手机 ...
- Spring_总结_04_高级配置(六)_Bean的初始化和销毁
一.前言 本文承接上一节:Spring_总结_04_高级配置(五)_运行时注入值
- BestCoder Round #18(hdu5105)Math Problem(高中数学)
最大值无非就是在两个端点或极值点处取得. 我注意讨论了a=0和b=0,却忽略了极值点可能不在L到R的范围内这一问题.被Hack了. #include<iostream> #include& ...
- file_put_contents();
file_put_contents(); 用于获取文件中的内容,可以填写网址,但是需要以http://开头
- [基本操作] kd 树
概念就不说了吧,网上教程满天飞 学了半天才知道,kd 树实质上只干了两件事情: 1.快速定位一个点 / 矩形 2.有理有据地优化暴力 第一点大概是可以来做二维平面上给点/矩形打标记的问题 第二点大概是 ...
- 服务端缓存页面及IIS缓存设置
缓存信息基本概念 我们在看网页的header信息时,经常看到这几个参数:Expires.Cache-Control.Last-Modified.ETag,它们是RFC 2616(HTTP/1.1)协议 ...
- 微信小程序 request域名配置好之后,还是提示报错配置的域名不在request合法域名中
自己尝试着用nodejs搭个后台服务的时候,用了端口号,然后在小程序中使用的时候,报错说配置的域名不在request合法域名中 明明已经配置好了的啊,看着报错信息.仔细对比了一下两个url请求地址,发 ...
- ShadowGun 图形技术分析
https://zhuanlan.zhihu.com/p/27966138 ShadowGun虽然是2011年的移动平台的游戏demo,但是里面的很多优化技巧到现在来看都是很值得学习的,毕竟是上过西瓜 ...
- [转]移动H5前端性能优化指南
移动H5前端性能优化指南 概述 1. PC优化手段在Mobile侧同样适用2. 在Mobile侧我们提出三秒种渲染完成首屏指标3. 基于第二点,首屏加载3秒完成或使用Loading4. 基于联通3G网 ...
- POJ1456:Supermarket(并查集版)
浅谈并查集:https://www.cnblogs.com/AKMer/p/10360090.html 题目传送门:http://poj.org/problem?id=1456 堆作法:https:/ ...