Compute real eigen values and eigen vectors of a symmetric matrix using QR decomposition method.
More...
#include <assert.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "qr_decompose.h"
|
#define | LIMS 9 |
| limit of range of matrix values
|
|
#define | EPSILON 1e-10 |
| accuracy tolerance limit
|
|
|
void | create_matrix (double **A, int N) |
| create a square matrix of given size with random elements More...
|
|
double ** | mat_mul (double **A, double **B, double **OUT, int R1, int C1, int R2, int C2) |
| Perform multiplication of two matrices. More...
|
|
double | eigen_values (double **A, double *eigen_vals, int mat_size, char debug_print) |
| Compute eigen values using iterative shifted QR decomposition algorithm as follows: More...
|
|
void | test1 () |
| test function to compute eigen values of a 2x2 matrix More...
|
|
void | test2 () |
| test function to compute eigen values of a 2x2 matrix More...
|
|
int | main (int argc, char **argv) |
| main function
|
|
Compute real eigen values and eigen vectors of a symmetric matrix using QR decomposition method.
- Author
- Krishna Vedala
◆ create_matrix()
void create_matrix |
( |
double ** |
A, |
|
|
int |
N |
|
) |
| |
create a square matrix of given size with random elements
- Parameters
-
[out] | A | matrix to create (must be pre-allocated in memory) |
[in] | N | matrix size |
29 int i, j, tmp, lim2 =
LIMS >> 1;
34 for (i = 0; i <
N; i++)
36 A[i][i] = (rand() %
LIMS) - lim2;
37 for (j = i + 1; j <
N; j++)
39 tmp = (rand() %
LIMS) - lim2;
◆ eigen_values()
double eigen_values |
( |
double ** |
A, |
|
|
double * |
eigen_vals, |
|
|
int |
mat_size, |
|
|
char |
debug_print |
|
) |
| |
Compute eigen values using iterative shifted QR decomposition algorithm as follows:
- Use last diagonal element of A as eigen value approximation \(c\)
- Shift diagonals of matrix \(A' = A - cI\)
- Decompose matrix \(A'=QR\)
- Compute next approximation \(A'_1 = RQ \)
- Shift diagonals back \(A_1 = A'_1 + cI\)
- Termination condition check: last element below diagonal is almost 0
- If not 0, go back to step 1 with the new approximation \(A_1\)
- If 0, continue to step 7
- Save last known \(c\) as the eigen value.
- Are all eigen values found?
- If not, remove last row and column of \(A_1\) and go back to step 1.
- If yes, stop.
- Note
- The matrix \(A\) gets modified
- Parameters
-
[in,out] | A | matrix to compute eigen values for |
[out] | eigen_vals | resultant vector containing computed eigen values |
[in] | mat_size | matrix size |
[in] | debug_print | 1 to print intermediate Q & R matrices, 0 for not to |
- Returns
- time for computation in seconds
107 double **R = (
double **)malloc(
sizeof(
double *) * mat_size);
108 double **Q = (
double **)malloc(
sizeof(
double *) * mat_size);
112 perror(
"Output eigen value vector cannot be NULL!");
117 perror(
"Unable to allocate memory for Q & R!");
122 for (
int i = 0; i < mat_size; i++)
124 R[i] = (
double *)malloc(
sizeof(
double) * mat_size);
125 Q[i] = (
double *)malloc(
sizeof(
double) * mat_size);
128 perror(
"Unable to allocate memory for Q & R.");
136 int rows = mat_size, columns = mat_size;
137 int counter = 0, num_eigs = rows - 1;
140 clock_t t1 = clock();
144 while (fabs(A[num_eigs][num_eigs - 1]) >
EPSILON)
146 last_eig = A[num_eigs][num_eigs];
147 for (
int i = 0; i < rows; i++) A[i][i] -= last_eig;
155 printf(
"-------------------- %d ---------------------\n",
159 mat_mul(R, Q, A, columns, columns, rows, columns);
160 for (
int i = 0; i < rows; i++) A[i][i] += last_eig;
164 eigen_vals[num_eigs] = last_eig;
168 printf(
"========================\n");
169 printf(
"Eigen value: % g,\n", last_eig);
170 printf(
"========================\n");
177 eigen_vals[0] = A[0][0];
178 double dtime = (double)(clock() - t1) / CLOCKS_PER_SEC;
187 for (
int i = 0; i < mat_size; i++)
◆ mat_mul()
double** mat_mul |
( |
double ** |
A, |
|
|
double ** |
B, |
|
|
double ** |
OUT, |
|
|
int |
R1, |
|
|
int |
C1, |
|
|
int |
R2, |
|
|
int |
C2 |
|
) |
| |
Perform multiplication of two matrices.
- R2 must be equal to C1
- Resultant matrix size should be R1xC2
- Parameters
-
[in] | A | first matrix to multiply |
[in] | B | second matrix to multiply |
[out] | OUT | output matrix (must be pre-allocated) |
[in] | R1 | number of rows of first matrix |
[in] | C1 | number of columns of first matrix |
[in] | R2 | number of rows of second matrix |
[in] | C2 | number of columns of second matrix |
- Returns
- pointer to resultant matrix
64 perror(
"Matrix dimensions mismatch!");
72 for (i = 0; i < R1; i++)
73 for (
int j = 0; j < C2; j++)
76 for (
int k = 0; k < C1; k++) OUT[i][j] += A[i][k] * B[k][j];
◆ test1()
test function to compute eigen values of a 2x2 matrix
\[\begin{bmatrix} 5 & 7\\ 7 & 11 \end{bmatrix}\]
which are approximately, {15.56158, 0.384227}
209 double X[][2] = {{5, 7}, {7, 11}};
210 double y[] = {15.56158, 0.384227};
214 double **A = (
double **)malloc(mat_size *
sizeof(
double *));
215 for (
int i = 0; i < mat_size; i++) A[i] = X[i];
217 printf(
"------- Test 1 -------\n");
221 for (
int i = 0; i < mat_size; i++)
223 printf(
"%d/5 Checking for %.3g --> ", i + 1, y[i]);
225 for (
int j = 0; j < mat_size && !result; j++)
227 if (fabs(y[i] - eig_vals[j]) < 0.1)
230 printf(
"(%.3g) ", eig_vals[j]);
238 printf(
"Test 1 Passed in %.3g sec\n\n", dtime);
◆ test2()
test function to compute eigen values of a 2x2 matrix
\[\begin{bmatrix} -4& 4& 2& 0& -3\\ 4& -4& 4& -3& -1\\ 2& 4& 4& 3& -3\\ 0& -3& 3& -1&-1\\ -3& -1& -3& -3& 0 \end{bmatrix}\]
which are approximately, {9.27648, -9.26948, 2.0181, -1.03516, -5.98994}
256 double X[][5] = {{-4, 4, 2, 0, -3},
260 {-3, -1, -3, -3, 0}};
261 double y[] = {9.27648, -9.26948, 2.0181, -1.03516,
266 double **A = (
double **)malloc(mat_size *
sizeof(
double *));
267 for (
int i = 0; i < mat_size; i++) A[i] = X[i];
269 printf(
"------- Test 2 -------\n");
273 for (
int i = 0; i < mat_size; i++)
275 printf(
"%d/5 Checking for %.3g --> ", i + 1, y[i]);
277 for (
int j = 0; j < mat_size && !result; j++)
279 if (fabs(y[i] - eig_vals[j]) < 0.1)
282 printf(
"(%.3g) ", eig_vals[j]);
290 printf(
"Test 2 Passed in %.3g sec\n\n", dtime);