eigen 列向量转矩阵_快速入门矩阵运算——开源库Eigen


矩阵是数学中一个重要的工具,广泛应用于各种场景下的数值分析,例如,数字信号处理,图像处理等。我们如何在程序中使用矩阵进行运算呢?本文将为大家介绍一个开源的矩阵运算工具——Eigen。

Eigen is a C++ template library for linear algebra: matrices, vectors, numerical solvers, and related algorithms.

Eigen是一个用于线性运算的C++ 模板库,支持 矩阵和矢量运算,数值分析及其相关的算法。

安装Eigen比较简单需要,从官网下载源码并解压即可,我现在的是最新的eigen-3.3.7版本。

官网下载地址:

http://eigen.tuxfamily.org/index.php?title=Main_Page

c723a6e832da5cf85b18da1e8a54fa79.png

我们可以进入Eigen目录,可以发现Eigen库主要包括如下几个模块组成:

  • Core:Matrix和Array类,基础的线性代数运算和数组操作;
  • Geometry:旋转,平移,缩放,2维和3维的各种变换;
  • LU:求逆,行列式,LU分解;
  • Cholesky:LLT和LDLT Cholesky分解;
  • Householder:Householder变换;
  • SVD:SVD分解;
  • QR:QR分解。
  • Eigenvalues:特征值,特征向量分解。
  • Sparse:稀疏矩阵的存储和运算。
  • Dense:包含了Core、Geometry、LU、Cholesky、SVD、QR、Eigenvalues等模块。
  • Eigen:包含了Dense和Sparse模块。
fa2203930ac90013e195fdef86a41c14.png

Eigen的食用方法非常之简单,在使用时我们只需要从解压后的文件目录中找到需要使用的库,然后,在源代码中包含相应的库即可食用了。因为Eigen是用模板写的模板库,所以只能把头文件包含进来使用。W君是在工程工作目录解压的,请参考如下代码包含Eigen库。

#include "eigen_3_3_7/Eigen/Eigen"

Matrix和Array模板类

Eigen库提供有Matrix和Array两种模板类。它们定义如下:

typedef Matrix MyMatrixType;typedef Array MyArrayType;

其中,通常我们会根据需要设置前三个参数,其它为默认值即可。

  • Scalar:指定元素类型,比如,float, double, bool, int 等。
  • RowsAtCompileTime:指定行数或者设置成动态(Dynamic);
  • ColsAtCompileTime:指定列数或者设置成动态(Dynamic);
  • Options:标志位,可以是ColMajor或RowMajor,默认是ColMajor;

从上面可以看出,行数和列数是允许固定大小,也允许动态大小的,所以下面的几种类型是可以的。

MatrixMatrixMatrixMatrixArrayArray

另外,我们还可以使用Eigen库已经重定义的类型,下面是一些简单的例子可参考。

372abfa847db3b8706ca12f5ab3cf98f.png

矩阵的定义和初始化

下面我们先看看Matrix模板类,它包含矩阵(Matrix)和向量(Vector),对于Vector来说,它也是一个矩阵,不过特殊的是行或者列数为1,我们称作行向量或者列向量。接下来我们来写个程序来看一下。

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::MatrixXf matrix1(3,4); //定义了矩阵的大小,但是没有初始化。    Eigen::Vector3f vector1;    matrix1 = Eigen::MatrixXf::Zero(3,4); //对矩阵进行初始化。    vector1 = Eigen::Vector3f::Ones();    std::cout << "------ matrix1 ------" << std::endl << matrix1 << std::endl;    std::cout << "------ vector1 ------" << std::endl << vector1 << std::endl;}

在上面的代码中,MatrixXf是一个行列可动态设置的矩阵,Vector3f是一个有3个元素的列向量。

typedef Matrix MatrixXf;typedef Matrix Vector3f;

需要注意到是在定义矩阵大小时是没有初始化矩阵的,需要重新对矩阵进行初始化,这里是用Zero和Ones函数对其初始化,Zero是初始化为全0,而Ones是初始化为全1。最后,我们看一下执行结果。

47bd9538e9ac9c3b0bf73434fd26e017.png

当然,我们也可以给定任意值,每个值用逗号隔开,不过,这种方法只能适用于确定矩阵大小的情况下使用。

    Eigen::MatrixXf matrix1(3,4);    Eigen::Vector3f vector1;    matrix1 << 1, 2,  3,  4,                5, 6,  7,  8,               9, 10, 11, 12;    vector1 << 1, 2, 3;

执行结果如下:

c8f58538e22842113415857ad684d969.png

另外,我们还可以用()对某个元素进行访问,我们可以对其可读和可写。

    Eigen::MatrixXf matrix1(3,4);    Eigen::Vector3f vector1;    matrix1 << 1, 2,  3,  4,                5, 6,  7,  8,               9, 10, 11, 12;    vector1 << 1, 2, 3;    matrix1(1,3) = 100;    vector1(1) = 100;

我们将matrix1的第2行,第4列修改成100;vector1的第2行修改成100,程序执行结果如下:

6dbe377847530b8bb9e254a3c6a86f33.png

矩阵的基本运算

学习了矩阵的定义,那我们下面来看一下几个基本的矩阵运算。Matrix重载了+,+=,-,-+,×,×=,/,/=这几个基本的四则运算。

对于加减运算,需要注意一下几点:

  • 左右矩阵的行列对应相等;
  • 数据的类型也要相同,因为矩阵运算不支持隐式类型转换;
  • 不支持和标量进行加减运算。

我们来看一个例子:

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::MatrixXf matrix1(2,3);    Eigen::MatrixXf matrix2(2,3);    matrix1 << 1, 2, 3,               5, 6, 7;    matrix2 << 1, 1, 2,               2, 1, 1;    std::cout << "------ matrix1 ------" << std::endl << matrix1 << std::endl;    std::cout << "------ matrix2 ------" << std::endl << matrix1 << std::endl;    std::cout << "------ matrix1 + matrix2 ------" << std::endl << matrix1 + matrix2 << std::endl;}

执行结果如下:

492517c7c984638b98253eb43c7364fc.png

关于乘除法运算,矩阵支持矩阵和标量之间的乘除法运算,标量和矩阵中的每个元素相运算。

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::MatrixXf matrix1(2,3);    Eigen::MatrixXf matrix2(2,3);    matrix1 << 1, 2, 3,               5, 6, 7;    matrix2 << 1, 1, 2,               2, 1, 1;    std::cout << "------ matrix1 ------" << std::endl << matrix1 << std::endl;    std::cout << "------ matrix2 ------" << std::endl << matrix2 << std::endl;    std::cout << "------ matrix1 * 2 ------" << std::endl << matrix1 * 2 << std::endl;    std::cout << "------ matrix2 / 2 ------" << std::endl << matrix2 / 2 << std::endl;}

执行结果如下:

42c9e4cfa2ee21b5a5ddb769c8d9bddb.png

关于矩阵和矩阵之间的乘法,两个矩阵只有当左边的矩阵的列数等于右边矩阵的行数时,两个矩阵才可以进行矩阵的乘法运算 。如下图,左边矩阵第i行的元素逐个和右边矩阵第j列的元素相乘,乘积再相加,就是第i行,第j列的结果。对于Eigen库来说,它通过重载*运算符帮我们实现了这个运算过程,我们直接使用*运算符即可。

99511d281482c2374582b406ebf383ef.png
#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::MatrixXf matrix1(2,3);    Eigen::MatrixXf matrix2(3,2);    matrix1 << 1, 2, 3,               5, 6, 7;    matrix2 << 1, 1,               2, 1,               3, 1;    std::cout << "------ matrix1 ------" << std::endl << matrix1 << std::endl;    std::cout << "------ matrix2 ------" << std::endl << matrix2 << std::endl;    std::cout << "------ matrix1 * matrix2 ------" << std::endl << matrix1 * matrix2 << std::endl;}

执行结果如下:

03eade6bf88b90a3c5ed0ddc7203c41a.png

另外,还包括矩阵的转置、共轭和共轭转置等矩阵运算,在Eigen中分别是有transpose()、conjugate()、adjoint()函数可实现。

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::MatrixXf matrix1(2, 3);    Eigen::MatrixXf matrix2(2, 3);    matrix1 << 1, 2, 3,               4, 5, 6;    std::cout << "------ matrix1 ------" << std::endl << matrix1 << std::endl;    // 转置    std::cout << "------ matrix1 transpose------" << std::endl << matrix1.transpose() << std::endl;    // 共轭    std::cout << "------ matrix1 conjugate------" << std::endl << matrix1.conjugate() << std::endl;    // 共轭转置    std::cout << "------ matrix1 adjoint------" << std::endl << matrix1.adjoint() << std::endl;}

执行结果如下,不过,代码中实数的共轭还是其本身。

93a689faf21781b1b868cb4ef8570f5a.png

数组的定义和初始化

有的时候我们并不需要做线性代数矩阵运算,而只是做元素级的操作,这个时候就需要用到Array了。对Array来说,它和Matrix的定义和初始化的方式是一样的。

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::ArrayXXf array1(2,3);    Eigen::ArrayXXf array2(3,3);    Eigen::ArrayXXf array3(5,10);    array1(0,0) = 1;    array1(0,1) = 2;    array1(0,2) = 2;    array1(1,0) = 1;    array1(1,1) = 2;    array1(1,2) = 2;    array2 << 1, 2, 3,              4, 5, 6,              7, 8, 9;    array3 = Eigen::ArrayXXf::Ones(5, 10);    std::cout << "------ array1 ------" << std::endl << array1 << std::endl;    std::cout << "------ array2 ------" << std::endl << array2 << std::endl;    std::cout << "------ array3 ------" << std::endl << array3 << std::endl;}

这里的ArrayXXf也是Eigen库重定义的,它行和列数可以动态设置。我们可以看到Array的初始化和Matrix也是类似的 。

typedef Array  ArrayXXf;

下面是执行结果:

d13c47ecd9b49c15bcc9a6aa25f489ee.png

数组的基本运算

Array同样也重载了四则运算,但是,这里数据只是对数组内的元素做算术运算,而不是矩阵线性运算。所以,对于Array来数,两个数据的大小相同,即行数和行数相同,列数和列数相同,它的四则运算就是取两个数组相应位置的元素参与运算,运算的结果就是对应位置的值。

当然,也支持和标量之间的运算,数组中的每个元素都乘以这个标量,这个和矩阵一样。

#include #include "eigen_3_3_7/Eigen/Eigen"int main(){    Eigen::ArrayXXf array1(2, 3);    Eigen::ArrayXXf array2(2, 3);    array1 << 1, 2, 3,               4, 5, 6;    array2 << 1, 2, 1,              1, 1, 2;    std::cout << "------ array1 ------" << std::endl << array1 << std::endl;    std::cout << "------ array2 ------" << std::endl << array2 << std::endl;    std::cout << "------ array1 + array2 ------" << std::endl << array1 + array2 << std::endl;    std::cout << "------ array1 - array2 ------" << std::endl << array1 - array2 << std::endl;    std::cout << "------ array1 * array2 ------" << std::endl << array1 * array2 << std::endl;    std::cout << "------ array1 / array2 ------" << std::endl << array1 / array2 << std::endl;    std::cout << "------ array1 * 2 ------" << std::endl << array1 * 2 << std::endl;}

执行结果:

8352c10b1974a8a64ef75fac588aa460.png

Matrix和Array两者是可以相互转换的。Matrix类有array()方法,可将Matrix转换为Array。Array类有matrix()方法,可将Array转换成Matrix。

总之,Eigen是一个很不错的开源库,官网还有更加详细的文档,大家可以参考:

http://eigen.tuxfamily.org/dox/GettingStarted.html

http://eigen.tuxfamily.org/dox/group__TutorialMatrixClass.html

http://eigen.tuxfamily.org/dox/group__QuickRefPage.html

http://eigen.tuxfamily.org/dox/Eigen2ToEigen3.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/459181.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql raid_DBA们应该知道的RAID卡知识_MySQL

bitsCN.com对于数据库这种特殊应用IOphotoshop/ target_blank classinfotextkey>PS往往会成为瓶颈&#xff0c;突破的这个瓶颈的有效方法不多&#xff0c;软件方面主要是读写分离&#xff0c;垂直拆分&#xff0c;分区表技术&#xff0c;cluster。硬件方面主要是raid&#x…

基于Maven的SSH框架搭建

2019独角兽企业重金招聘Python工程师标准>>> 1.工程介绍 工程是结合了Springstruts2hibernate&#xff0c;实现了一个简单的form表单提交的功能&#xff0c;可能需要对spring&#xff0c;struts2&#xff0c;hibernate有一个基础的了解才好理解。 2.工程结构图 首先…

交通警察手势信号(动画演示)

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** 一、交通警察手势信号&#xff0d;停止信号 二、交通警察手势信号&#xff0d;直行信…

mysql和mybatis面试题_BATJ面试题汇总详解:MyBatis+MySQL+Spring+Redis+多线程

SpringSpring 概述什么是spring?使用Spring框架的好处是什么&#xff1f;Spring由哪些模块组成&#xff1f;解释AOP模块Spring配置文件什么是Spring IOC 容器&#xff1f;依赖注入什么是Spring的依赖注入&#xff1f;有哪些不同类型的IOC(依赖注入)方式&#xff1f;哪种依赖注…

Codeblocks和gdb调试 (转)

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** 使用C::B和gdb调试是一件简单的事情。下面&#xff0c;让我们调试一个简单的循环&…

mysql5.7.17 win7_win7下mysql5.7.17安装配置方法图文教程

win7下安装mysql5.7.17图文教程&#xff0c;分享给大家。1.下载安装包请在以下有zip包和msi两种类型包&#xff0c;建议新手选择zip包下载&#xff0c;有助于熟习mysql2.解压mysql压缩包下载完成后解压&#xff0c;将其放在要安装的目录下面&#xff0c;如&#xff1a;e:\mysql…

停一下

15年过去已半载有余&#xff0c;回头看年初定下的目标&#xff0c;有种管中窥豹的感觉。之前和肉山讨论的时候&#xff0c;他对我想要发展的方向并没有表示赞同。 现在认为他是对的&#xff0c;发展的方向太靠前了&#xff0c;ui&#xff0c;canvas&#xff0c;svg&#xff0c;…

which 命令

我们经常在linux要查找某个文件&#xff0c;但不知道放在哪里了&#xff0c;可以使用下面的一些命令来搜索&#xff1a; which 查看可执行文件的位置。 whereis 查看文件的位置。 locate 配合数据库查看文件位置。 find 实际搜寻硬盘查询文件名…

18ch

18.2 线程和进程 18.2.1 什么是进程&#xff1f; 18.2.1 什么是进程&#xff1f; 计算机程序只不过是磁盘中可执行的&#xff0c;二进制的数据。它们只有在被读取到内存中&#xff0c;被操作系统调用的时候才开始它们的生命周期。进程&#xff08;重量级进程&#xff09;是程序…

安卓四大组件总览

在安卓中四大组件 &#xff08;Component&#xff09;指的是&#xff1a;Activity&#xff0c;Service&#xff0c;BroadcastReceiver&#xff0c;ContentProvider。此博客仅仅对安卓中四大组件从整体上进行简单的分析&#xff0c;了解他们在安卓系统框架中处的位置与作用&…

java ee 指南 pdf_Java EE 7权威指南:卷1(原书第5版) 中文pdf

资源名称&#xff1a;Java EE 7权威指南&#xff1a;卷1(原书第5版) 中文pdf第一部分 引言第1章 概述 2第2章 使用教程示例 27第二部分 平台基础知识第3章 资源创建 38第4章 注入 41第5章 打包 44第三部分 Web层第6章 Web应用入门 50第7章 JSF技术 66第8章 Facelets…

PYTHON招聘需求与技能体系

为什么80%的码农都做不了架构师&#xff1f;>>> 目前国内的招聘Python&#xff0c;基本都是偏向web后台开发&#xff0c;偶有高大上的数据挖掘&机器学习 这是之前(2012年)找工作整理的一些JD&#xff0c;在梳理几年来的笔记&#xff0c;顺带理一理 可以以此建…

《FPGA全程进阶---实战演练》第二十一章 电源常用类型:LDO和 DCDC

高速电路中的电源设计 高速电路中的电源设计大概分为两种&#xff0c;一种是集总式架构&#xff0c;一种是分布式架构。集总式架构就是由一个电源输入&#xff0c;然后生成多种所需要的电压。如图1所示。这种架构会增加多个DC/DC模块&#xff0c;这样成本不可控&#xff0c;PCB…

迁云架构实践

本文着笔介绍IT互联网化为传统企业带来的技术挑战&#xff0c;并对上云架构最佳实践进行了深入介绍&#xff0c;首发于阿里云&《程序员》联合出品的《凌云》杂志。 作者&#xff1a; 王宇德&#xff0c;张文生 云计算作为信息技术领域的一种创新应用模式&#xff0c;自其诞…

最短路最新心得

如果&#xff0c;上面的图&#xff0c;如果用dij算法&#xff0c;那么dist[4] 4, 是得不到正确的结果的&#xff0c; 这个因为dist[3]先被确定是最小&#xff0c;然后用来更新dist[4] 但是存在负权&#xff0c;使得dist[3]更小&#xff0c;但是我们已经把结点3标记为不可用了…

java wrapper怎么运行_如何从智能合约中生成Java Wrapper

在本文中&#xff0c;我们将了解如何直接从智能合约生成Java Wrapper类以与Java中的智能合约进行交互。从智能合约生成Java Wrapper类有不同的方法&#xff1a;1. Web3j命令行工具和solc2. Web3j命令行工具和Truffle构建生成的工件3. web3j-maven-plugin4. web3j-gradle-plugin…

Matlab与C/C++混合编程调用OpenCV

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com http://item.taobao.com/item.htm?spma1z10.5-c.w4002-9510581626.24.ZO6sko&id43401674106 精通MATLAB混合编程视频讲解 MATLAB各类函数…

python 程序打包 vscode_使用VScode编写python程序并打包成.exe文件

听说Visual Studio Code(VS Code)的诸多好处&#xff0c;了解了一下果真很喜欢&#xff0c;我喜欢它的缘由主要有3个&#xff0c;一是VS Code开源且跨平台&#xff0c;二是由于其界面很是酷&#xff0c;三是能够知足个人大所属代码需求&#xff0c;除此以外固然还有强大的好奇心…

使用命名空间、头文件和实现文件

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** 2.3 使用命名空间、头文件和实现文件 使新的throttle类满足程序的需求将是非常…

Permissions for id_rsa are too open

为什么80%的码农都做不了架构师&#xff1f;>>> Last week I was lucky enough to have my laptop upgraded (yay SSD) which meant I needed to configure a new install of OSX Lion. Most of my files are stored online (email, documents, code, etc) but I m…