This forum has been archived. All content is frozen. Please use KDE Discuss instead.

LU module incorrectly claims that matrix is not invertible

Tags: None
(comma "," separated)
bgamari
Registered Member
Posts
6
Karma
0
I was recently quite surprised when I found that the bug I was chasing appears to be the result of Eigen's LU decomposition algorithm incorrectly claiming that my matrix is non-invertible. I have attached the matrix (R) as well as a short piece of code (test.cpp) to reproduce this issue. On my machine (GCC 4.4, Eigen 2.0.12), the reproduction code fails with,

$ ./test
Error: Not invertible

Despite the fact that Octave has no difficulty inverting the same matrix,

octave:1> load "R"
octave:2> rank(transpose(R)*R)
ans = 10
octave:3> inv(transpose(R)*R)
ans = ...

Note that I am inverting transpose(R)*R, not R. I have verified that Eigen computes this correctly, however.

Any help you could offer would be greatly appreciated. Thanks!

- Ben
bgamari
Registered Member
Posts
6
Karma
0
Apparently it's not possible to attach files here. I have posted the reproduction code and associated data at http://goldnerlab.physics.umass.edu/~bg ... est_eigen/ .

Thanks again,

- Ben
User avatar
bjacob
Registered Member
Posts
658
Karma
3
This problem, which is inherent to the Eigen 2.0 API, has been fixed in Eigen 3 (the development branch). I encourage you to move to it. Does it fix your problem?


Join us on Eigen's IRC channel: #eigen on irc.freenode.net
Have a serious interest in Eigen? Then join the mailing list!
bgamari
Registered Member
Posts
6
Karma
0
Sadly, Eigen 3 does not fix the issue. I have uploaded a new test program based on Eigen 3, yet the matrix is still claimed to be uninvertible.

- Ben
User avatar
bjacob
Registered Member
Posts
658
Karma
3
This is not a bug, your RR matrix is actually not invertible to machine precision, with single-precision floats!

However,
- the matrix R is almost (but not quite) invertible to machine precision with float;
- the matrix RR would be almost (but not quite) invertible to machine precision if you were using double precision.

Probably Octave is using double precision, and is being lucky. You can do that with Eigen3 too, just use PartialPivLU or use FullPivLU with setThreshold(0). Not recommended, anyway. There's almost no good reason why you'd want a LU decomposition of that kind of matrix (as opposed to other decompositions).

Here's how I found out:
Code: Select all
--- /home/bjacob/Downloads/test.cpp     2010-05-25 23:43:01.266590944 -0400
+++ test.cpp    2010-05-25 23:41:47.130349108 -0400
@@ -1,7 +1,6 @@
 #include <iostream>
 #include <fstream>
-#include <Eigen/Core>
-#include <Eigen/LU>
+#include <Eigen/Eigen>
 
 using namespace Eigen;
 
@@ -25,6 +24,10 @@
         }
         dump_matrix(R, "R.test");       
         Matrix<float, 10,10> RR = R.transpose() * R;
+        JacobiSVD<Matrix<float, 10,10> > svd_of_R(R);
+        std::cout << "singular values of R:\n" << svd_of_R.singularValues() << std::endl;
+        JacobiSVD<Matrix<float, 10,10> > svd_of_RR(RR);
+        std::cout << "singular values of RR:\n" << svd_of_RR.singularValues() << std::endl;
         dump_matrix(RR, "RR.test");
         FullPivLU<Matrix<float, 10,10> > lu(RR);
         if (!lu.isInvertible()) {


result:
Code: Select all
##### 23:41:53 ~/cuisine$ ./test
singular values of R:
12.9921
0.0675159
0.0284538
0.00920528
0.000302491
8.80375e-05
2.89062e-05
1.08431e-05
4.84466e-06
5.91241e-07
singular values of RR:
168.795
0.00455854
0.000808845
8.97432e-05
4.36866e-06
3.72895e-07
9.01673e-08
7.15757e-08
3.72439e-08
1.4729e-08
Error: Not invertible


Join us on Eigen's IRC channel: #eigen on irc.freenode.net
Have a serious interest in Eigen? Then join the mailing list!
bgamari
Registered Member
Posts
6
Karma
0
Hmmm. Well, if you couldn't tell my linear algebra knowledge is sorely lacking. This code is ultimately trying to produce a linear regression (ordinary least squares with terms up to second order) for data from a set of sensors. If I plot the data, it looks pretty planar but the data certainly isn't free of noise. Could this perhaps be the reason for the singular values? Otherwise, it seems very strange to me that a matrix formed from a fairly well-behaved data-set would be singular.

It appears that the invertible matrix theorem dictates that as long as the columns of R are linearly independent RR should be invertible[1].


[1] http://books.google.com/books?id=MjNv6r ... &q&f=false
bgamari
Registered Member
Posts
6
Karma
0
After doing some research, it seems that my problem may be the overdetermined nature of my regression. It seems that using singular value decomposition is a pretty common approach for doing these sorts of regressions. Unfortunately, I don't see how one would use Eigen's SVD module actually compute the components of the decomposition. Both Eigen::JacobiSVD and Eigen::SVD have solve() functions, but there is nothing else to suggest that one can access the underlying decomposition (unless I'm missing something).

Any advice? Thanks again.
User avatar
bjacob
Registered Member
Posts
658
Karma
3
i confirm that SVD is what you want here, and rather JacobiSVD. The components of the SVD decomposition are the U and V matrices (aka the singular vectors), and the singular values. All of them can be obtained by methods with self-explanatory names in class JacobiSVD. Note that the API of JacobiSVD is going to get simplified very soon so don't be surprised if code written today doesn't compile anymore in 1 week. You'll have to adapt only once.

Note: you'll need only one among U and V, make sure you skip the other one (SkipU/SkipV option to jacobiSVD), or else it's going to be slow.


Join us on Eigen's IRC channel: #eigen on irc.freenode.net
Have a serious interest in Eigen? Then join the mailing list!
bgamari
Registered Member
Posts
6
Karma
0
Thanks a ton for your help. I'm now simply using SVD::solve to solve the linear system directly. Is there a reason the SVD::matrix*() functions aren't documented? Thanks again!

- Ben
User avatar
bjacob
Registered Member
Posts
658
Karma
3
The development branch's documentation is lacking, we know, will fix this before the release.

solve() is ... a nice first step as long as it's working, but eventually, when a singular value will be exactly 0, it won't work anymore. The real solution is to read the wikipedia page on least squares using the SVD... anyway, good luck!


Join us on Eigen's IRC channel: #eigen on irc.freenode.net
Have a serious interest in Eigen? Then join the mailing list!


Bookmarks



Who is online

Registered users: Bing [Bot], Google [Bot], Yahoo [Bot]