2023-05-23 05:42:20 +00:00
|
|
|
//! Data structures and traits for decomposing and solving matrices
|
|
|
|
|
2023-05-06 07:39:06 +00:00
|
|
|
#[macro_use]
|
|
|
|
mod common;
|
|
|
|
|
|
|
|
use common::Approx;
|
|
|
|
use generic_parameterize::parameterize;
|
|
|
|
use num_traits::real::Real;
|
2023-05-22 05:52:47 +00:00
|
|
|
use num_traits::Zero;
|
2023-05-06 07:39:06 +00:00
|
|
|
use std::fmt::Debug;
|
2023-05-22 03:55:37 +00:00
|
|
|
use vector_victor::decompose::{LUDecompose, LUDecomposition, Parity};
|
2023-05-06 08:34:31 +00:00
|
|
|
use vector_victor::{Matrix, Vector};
|
2023-05-06 07:39:06 +00:00
|
|
|
|
|
|
|
#[parameterize(S = (f32, f64), M = [1,2,3,4])]
|
|
|
|
#[test]
|
|
|
|
/// The LU decomposition of the identity matrix should produce
|
|
|
|
/// the identity matrix with no permutations and parity 1
|
2023-05-22 03:55:37 +00:00
|
|
|
fn test_lu_identity<S, const M: usize>()
|
|
|
|
where
|
|
|
|
Matrix<S, M, M>: LUDecompose<S, M>,
|
|
|
|
S: Copy + Real + Debug + Approx + Default,
|
|
|
|
{
|
2023-05-06 07:39:06 +00:00
|
|
|
// let a: Matrix<f32, 3, 3> = Matrix::<f32, 3, 3>::identity();
|
|
|
|
let i = Matrix::<S, M, M>::identity();
|
|
|
|
let ones = Vector::<S, M>::fill(S::one());
|
|
|
|
let decomp = i.lu().expect("Singular matrix encountered");
|
2023-05-07 06:14:43 +00:00
|
|
|
let LUDecomposition { lu, idx, parity } = decomp;
|
2023-05-06 07:39:06 +00:00
|
|
|
assert_eq!(lu, i, "Incorrect LU decomposition");
|
|
|
|
assert!(
|
|
|
|
(0..M).eq(idx.elements().cloned()),
|
|
|
|
"Incorrect permutation matrix",
|
|
|
|
);
|
2023-05-22 03:55:37 +00:00
|
|
|
assert_eq!(parity, Parity::Even, "Incorrect permutation parity");
|
2023-05-06 07:39:06 +00:00
|
|
|
|
|
|
|
// Check determinant calculation which uses LU decomposition
|
|
|
|
assert_approx!(
|
|
|
|
i.det(),
|
|
|
|
S::one(),
|
|
|
|
"Identity matrix should have determinant of 1"
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check inverse calculation with uses LU decomposition
|
|
|
|
assert_eq!(
|
2023-05-22 03:55:37 +00:00
|
|
|
i.inv(),
|
2023-05-06 07:39:06 +00:00
|
|
|
Some(i),
|
|
|
|
"Identity matrix should be its own inverse"
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
i.solve(&ones),
|
|
|
|
Some(ones),
|
|
|
|
"Failed to solve using identity matrix"
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check triangle separation
|
|
|
|
assert_eq!(decomp.separate(), (i, i));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[parameterize(S = (f32, f64), M = [2,3,4])]
|
|
|
|
#[test]
|
|
|
|
/// The LU decomposition of any singular matrix should be `None`
|
2023-05-22 03:55:37 +00:00
|
|
|
fn test_lu_singular<S, const M: usize>()
|
|
|
|
where
|
|
|
|
Matrix<S, M, M>: LUDecompose<S, M>,
|
|
|
|
S: Copy + Real + Debug + Approx + Default,
|
|
|
|
{
|
2023-05-06 07:39:06 +00:00
|
|
|
// let a: Matrix<f32, 3, 3> = Matrix::<f32, 3, 3>::identity();
|
|
|
|
let mut a = Matrix::<S, M, M>::zero();
|
|
|
|
let ones = Vector::<S, M>::fill(S::one());
|
|
|
|
a.set_row(0, &ones);
|
|
|
|
|
|
|
|
assert_eq!(a.lu(), None, "Matrix should be singular");
|
|
|
|
assert_eq!(
|
|
|
|
a.det(),
|
|
|
|
S::zero(),
|
|
|
|
"Singular matrix should have determinant of zero"
|
|
|
|
);
|
2023-05-22 03:55:37 +00:00
|
|
|
assert_eq!(a.inv(), None, "Singular matrix should have no inverse");
|
2023-05-06 07:39:06 +00:00
|
|
|
assert_eq!(
|
|
|
|
a.solve(&ones),
|
|
|
|
None,
|
|
|
|
"Singular matrix should not be solvable"
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_lu_2x2() {
|
2023-05-22 03:55:37 +00:00
|
|
|
let a = Matrix::mat([[1.0, 2.0], [3.0, 0.0]]);
|
2023-05-06 07:39:06 +00:00
|
|
|
let decomp = a.lu().expect("Singular matrix encountered");
|
|
|
|
// the decomposition is non-unique, due to the combination of lu and idx.
|
|
|
|
// Instead of checking the exact value, we only check the results.
|
|
|
|
// Also check if they produce the same results with both methods, since the
|
|
|
|
// Matrix<> methods use shortcuts the decomposition methods don't
|
|
|
|
|
|
|
|
let (l, u) = decomp.separate();
|
2023-05-06 08:34:31 +00:00
|
|
|
assert_approx!(l.mmul(&u), a.permute_rows(&decomp.idx));
|
2023-05-06 07:39:06 +00:00
|
|
|
|
|
|
|
assert_approx!(a.det(), -6.0);
|
|
|
|
assert_approx!(a.det(), decomp.det());
|
|
|
|
|
|
|
|
assert_approx!(
|
2023-05-22 03:55:37 +00:00
|
|
|
a.inv().unwrap(),
|
|
|
|
Matrix::mat([[0.0, 2.0], [3.0, -1.0]]) * (1.0 / 6.0)
|
2023-05-06 07:39:06 +00:00
|
|
|
);
|
2023-05-22 03:55:37 +00:00
|
|
|
assert_approx!(a.inv().unwrap(), decomp.inv());
|
|
|
|
assert_approx!(a.inv().unwrap().inv().unwrap(), a)
|
2023-05-06 07:39:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_lu_3x3() {
|
2023-05-22 03:55:37 +00:00
|
|
|
let a = Matrix::mat([[1.0, -5.0, 8.0], [1.0, -2.0, 1.0], [2.0, -1.0, -4.0]]);
|
2023-05-06 07:39:06 +00:00
|
|
|
let decomp = a.lu().expect("Singular matrix encountered");
|
|
|
|
|
|
|
|
let (l, u) = decomp.separate();
|
2023-05-06 08:34:31 +00:00
|
|
|
assert_approx!(l.mmul(&u), a.permute_rows(&decomp.idx));
|
2023-05-06 07:39:06 +00:00
|
|
|
|
|
|
|
assert_approx!(a.det(), 3.0);
|
|
|
|
assert_approx!(a.det(), decomp.det());
|
|
|
|
|
|
|
|
assert_approx!(
|
2023-05-22 03:55:37 +00:00
|
|
|
a.inv().unwrap(),
|
|
|
|
Matrix::mat([[9.0, -28.0, 11.0], [6.0, -20.0, 7.0], [3.0, -9.0, 3.0]]) * (1.0 / 3.0)
|
2023-05-06 07:39:06 +00:00
|
|
|
);
|
2023-05-22 03:55:37 +00:00
|
|
|
assert_approx!(a.inv().unwrap(), decomp.inv());
|
|
|
|
assert_approx!(a.inv().unwrap().inv().unwrap(), a)
|
2023-05-06 07:39:06 +00:00
|
|
|
}
|