Eigen:矩陣計算簡單用法(二)
【原文:http://blog.sina.com.cn/s/blog_691fc8920102v02q.html】
6、如何選擇動態矩陣和靜態矩陣?
還需特別注意的是:如果特別大的矩陣使用了固定大小的靜態矩陣則可能造成棧溢位的問題
---------------------------------------------------------------------------------------------
本文主要是Eigen中矩陣和向量的算術運算,在Eigen中的這些算術運算過載了C++的+,-,*,所以使用起來非常方便。
1、矩陣的運算
Eigen提供+、-、一元操作符“-”、+=、-=,例如:
二元操作符+/-表示兩矩陣相加(矩陣中對應元素相加/減,返回一個臨時矩陣): B+C 或 B-C;
一元操作符-表示對矩陣取負(矩陣中對應元素取負,返回一個臨時矩陣): -C;
組合操作法+=或者-=表示(對應每隔元素都做相應操作):A += B 或者 A-=B
程式碼段1為矩陣的加減操作,程式碼如下:
-
- #include
- #include
- using namespace Eigen;
- int main()
- {
- Matrix2d a;
- a << 1, 2,
- 3, 4;
- MatrixXd b(2,2);
- b << 2, 3,
- 1, 4;
- std::cout << "a + b =\n" << a + b << std::endl;
- std::cout << "a - b =\n" << a - b << std::endl;
- std::cout << "Doing a += b;" << std::endl;
- a += b;
- std::cout << "Now a =\n" << a << std::endl;
- Vector3d v(1,2,3);
- Vector3d w(1,0,0);
- std::cout << "-v + w - v =\n" << -v + w - v << std::endl;
- }
a + b = 3 5 4 8 a - b = -1 -1 2 0 Doing a += b; Now a = 3 5 4 8 -v + w - v = -1 -4 -6
另外,矩陣還提供與標量(單一個數字)的乘除操作,表示每個元素都與該標量進行乘除操作。例如:
二元操作符*在:A*a中表示矩陣A中的每隔元素都與數字a相乘,結果放在一個臨時矩陣中,矩陣的值不會改變。
對於a*A、A/a、A*=a、A /=a也是一樣,例如下面的程式碼:
- #include
- #include
- using namespace Eigen;
- int main()
- {
- Matrix2d a;
- a << 1, 2,
- 3, 4;
- Vector3d v(1,2,3);
- std::cout << "a * 2.5 =\n" << a * 2.5 << std::endl;
- std::cout << "0.1 * v =\n" << 0.1 * v << std::endl;
- std::cout << "Doing v *= 2;" << std::endl;
- v *= 2;
- std::cout << "Now v =\n" << v << std::endl;
- }
a * 2.5 = 2.5 5 7.5 10 0.1 * v = 0.1 0.2 0.3 Doing v *= 2; Now v = 2 4 6
需要注意:
在Eigen中,算術操作例如 “操作符+”並不會自己執行計算操作,他們只是返回一個“算術表示式物件”,而實際的計算則會延遲到後面的賦值時才進行。這些不影響你的使用,它只是為了方便Eigen的優化。
2、求矩陣的轉秩、共軛矩陣、伴隨矩陣。
可以通過 成員函式transpose(), conjugate(),和 adjoint()來完成,注意這些函式返回操作後的結果,而不會對原矩陣的元素進行直接操作,如果要讓原矩陣的進行轉換,則需要使用響應的InPlace函式,例如:transposeInPlace() 、 adjointInPlace() 之類。
例如下面的程式碼所示:
- MatrixXcf a = MatrixXcf::Random(2,2);
- cout << "Here is the matrix a\n" << a << endl;
- cout << "Here is the matrix a^T\n" << a.transpose() << endl;
- cout << "Here is the conjugate of a\n" << a.conjugate() << endl;
- cout << "Here is the matrix a^*\n" << a.adjoint() << endl;
Here is the matrix a (-0.211,0.68) (-0.605,0.823) (0.597,0.566) (0.536,-0.33) Here is the matrix a^T (-0.211,0.68) (0.597,0.566) (-0.605,0.823) (0.536,-0.33) Here is the conjugate of a (-0.211,-0.68) (-0.605,-0.823) (0.597,-0.566) (0.536,0.33) Here is the matrix a^* (-0.211,-0.68) (0.597,-0.566) (-0.605,-0.823) (0.536,0.33)
矩陣的相乘,矩陣與向量的相乘也是使用操作符*,共有*和*=兩種操作符,其用法可以參考如下程式碼:
- #include
- #include
- using namespace Eigen;
- int main()
- {
- Matrix2d mat;
- mat << 1, 2,
- 3, 4;
- Vector2d u(-1,1), v(2,0);
- std::cout << "Here is mat*mat:\n" << mat*mat << std::endl;
- std::cout << "Here is mat*u:\n" << mat*u << std::endl;
- std::cout << "Here is u^T*mat:\n" << u.transpose()*mat << std::endl;
- std::cout << "Here is u^T*v:\n" << u.transpose()*v << std::endl;
- std::cout << "Here is u*v^T:\n" << u*v.transpose() << std::endl;
- std::cout << "Let's multiply mat by itself" << std::endl;
- mat = mat*mat;
- std::cout << "Now mat is mat:\n" << mat << std::endl;
- }
Here is mat*mat: 7 10 15 22 Here is mat*u: 1 1 Here is u^T*mat: 2 2 Here is u^T*v: -2 Here is u*v^T: -2 -0 2 0 Let's multiply mat by itself Now mat is mat: 7 10 15 22--------------------------------------------------------------------------------------------
本節主要涉及Eigen的塊操作以及QR分解,Eigen的QR分解非常繞人,搞了很久才搞明白是怎麼回事,最後是一個使用Eigen的矩陣操作完成二維高斯擬合求取光點的程式碼例子,關於二維高斯擬合求取光點的詳細內容可參考:http://blog.csdn.net/hjx_1000/article/details/8490653
1、矩陣的塊操作
1)矩陣的塊操作有兩種使用方法,其定義形式為:
- matrix.block(i,j,p,q); (1)
-
matrix.block
(i,j); (2)
定義(2)中block(p, q)可理解為一個p行q列的子矩陣,該定義表示從原矩陣中第(i, j)開始,獲取一個p行q列的子矩陣,返回該子矩陣組成的臨時矩陣物件,原矩陣的元素不變。
詳細使用情況,可參考下面的程式碼段:
- #include
- #include
- using namespace std;
- int main()
- {
- Eigen::MatrixXf m(4,4);
- m << 1, 2, 3, 4,
- 5, 6, 7, 8,
- 9,10,11,12,
- 13,14,15,16;
- cout << "Block in the middle" << endl;
- cout << m.block<2,2>(1,1) << endl << endl;
- for (int i = 1; i <= 3; ++i)
- {
- cout << "Block of size " << i << "x" << i << endl;
- cout << m.block(0,0,i,i) << endl << endl;
- }
- }
Block in the middle 6 7 10 11 Block of size 1x1 1 Block of size 2x2 1 2 5 6 Block of size 3x3 1 2 3 5 6 7 9 10 11
通過上述方式獲取的子矩陣即可以作為左值也可以作為右值,也就是即可以用這個子矩陣給其他矩陣賦值,也可以給這個子矩陣物件賦值。
2)矩陣也提供了獲取其指定行/列的函式,其實獲取某行/列也是一種特殊的獲取子塊。可以通過 .col()和 .row()來完成獲取指定列/行的操作,引數為列/行的索引。
注意:
(1)需與獲取矩陣的行數/列數的函式( rows(), cols() )的進行區別,不要弄混淆。
(2)函式引數為響應行/列的索引,需注意矩陣的行列均以0開始。
下面的程式碼段用於演示獲取矩陣的指定行列:
- #include
- #include
- using namespace std;
- int main()
- {
- Eigen::MatrixXf m(3,3);
- m << 1,2,3,
- 4,5,6,
- 7,8,9;
- cout << "Here is the matrix m:" << endl << m << endl;
- cout << "2nd Row: " << m.row(1) << endl;
- m.col(2) += 3 * m.col(0);
- cout << "After adding 3 times the first column into the third column, the matrix m is:\n";
- cout << m << endl;
- }
Here is the matrix m: 1 2 3 4 5 6 7 8 9 2nd Row: 4 5 6 After adding 3 times the first column into the third column, the matrix m is: 1 2 6 4 5 18 7 8 30
3)向量的塊操作,其實向量只是一個特殊的矩陣,但是Eigen也為它單獨提供了一些簡化的塊操作,如下三種形式:
獲取向量的前n個元素:vector.head(n);
獲取向量尾部的n個元素:vector.tail(n);
獲取從向量的第i個元素開始的n個元素:vector.segment(i,n);
其用法可參考如下程式碼段:
- #include
- #include
- using namespace std;
- int main()
- {
- Eigen::ArrayXf v(6);
- v << 1, 2, 3, 4, 5, 6;
- cout << "v.head(3) =" << endl << v.head(3) << endl << endl;
- cout << "v.tail<3>() = " << endl << v.tail<3>() << endl << endl;
- v.segment(1,4) *= 2;
- cout << "after 'v.segment(1,4) *= 2', v =" << endl << v << endl;
- }
v.head(3) = 1 2 3 v.tail<3>() = 4 5 6 after 'v.segment(1,4) *= 2', v = 1 4 6 8 10 6
2、QR分解
Eigen的QR分解非常繞人,它總共提供了下面這些矩陣的分解方式:
Decomposition | Method | Requirements on the matrix | Speed | Accuracy |
---|---|---|---|---|
PartialPivLU | partialPivLu() | Invertible | ++ | + |
FullPivLU | fullPivLu() | None | - | +++ |
HouseholderQR | householderQr() | None | ++ | + |
ColPivHouseholderQR | colPivHouseholderQr() | None | + | ++ |
FullPivHouseholderQR | fullPivHouseholderQr() | None | - | +++ |
LLT | llt() | Positive definite | +++ | + |
LDLT | ldlt() | Positive or negative semidefinite | +++ | ++ |
由於我只用到了QR分解,而且Eigen的QR分解開始使用時確實不容易入手,因此這裡只提供了householderQR的分解方式的演示程式碼:
- void QR2()
- {
- Matrix3d A;
- A<<1,1,1,
- 2,-1,-1,
- 2,-4,5;
- HouseholderQR qr;
- qr.compute(A);
- MatrixXd R = qr.matrixQR().triangularView();
- MatrixXd Q = qr.householderQ();
- std::cout << "QR2(): HouseholderQR---------------------------------------------"<< std::endl;
- std::cout << "A "<< std::endl <<A << std::endl << std::endl;
- std::cout <<"qr.matrixQR()"<< std::endl << qr.matrixQR() << std::endl << std::endl;
- std::cout << "R"<< std::endl <<R << std::endl << std::endl;
- std::cout << "Q "<< std::endl <<Q << std::endl << std::endl;
- std::cout <<"Q*R" << std::endl <<Q*R << std::endl << std::endl;
- }
3、一個矩陣使用的例子:用矩陣操作完成二維高斯擬合,並求取光斑中心
下面的程式碼段是一個使用Eigen的矩陣操作完成二維高斯擬合求取光點的程式碼例子,關於二維高斯擬合求取光點的詳細內容可參考:http://blog.csdn.net/hjx_1000/article/details/8490653
http://blog.csdn.net/houjixin/article/details/8490941
http://blog.csdn.net/houjixin/article/details/8492841
http://blog.csdn.net/houjixin/article/details/8494582
相關文章
- Eigen:矩陣計算簡單用法(一)矩陣
- Eigen教程(3)之矩陣和向量的運算矩陣
- 矩陣計算矩陣
- 計算矩陣的秩矩陣
- 矩陣:如何使用矩陣操作進行 PageRank 計算?矩陣
- 矩陣旋轉-Eigen應用(QTCreator編輯器)矩陣QT
- 簡單的傳球遊戲(矩陣)遊戲矩陣
- 協方差矩陣概念及計算矩陣
- 矩陣中最大的二維矩陣矩陣
- 單像空間後方交會計算旋轉矩陣矩陣
- 矩陣乘法的運算量計算(華為OJ)矩陣
- 03:計算矩陣邊緣元素之和矩陣
- OpenCV矩陣運算OpenCV矩陣
- Wannafly模擬賽 矩陣 二維矩陣hash矩陣
- 矩陣求導(二)矩陣求導
- 怎樣用python計算矩陣乘法?Python矩陣
- matlab計算含有未知數的矩陣Matlab矩陣
- 深度學習中需要的矩陣計算深度學習矩陣
- 計算機演算法:Strassen矩陣乘法計算機演算法矩陣
- C++ 練氣期之二維陣列與矩陣運算C++陣列矩陣
- 什麼是新媒體矩陣運營?運營矩陣其實很簡單矩陣
- Julia的矩陣運算矩陣
- 計算機圖形學之矩陣變換計算機矩陣
- NYOJ 1409 快速計算【矩陣連乘】矩陣
- [WebGL入門]十,矩陣計算和外部庫Web矩陣
- 單像空間後方交會計算誤差方程係數矩陣矩陣
- 二維陣列的計算陣列
- Numpy中的矩陣運算矩陣
- 樣本協方差矩陣的定義與計算矩陣
- python 計算矩陣的相關演算法Python矩陣演算法
- 240. 搜尋二維矩陣 II 和74. 搜尋二維矩陣矩陣
- 資料結構之陣列和矩陣--矩陣&不規則二維陣列資料結構陣列矩陣
- Python numpy中矩陣的用法總結Python矩陣
- 簡單版計算器
- MFC簡單計算器
- PHP簡單計算器PHP
- Groovy簡單公式計算公式
- OpenGL光照計算中法線矩陣原理及推到過程矩陣