2017-08-03 01:37:44 +08:00
|
|
|
use alga::general::Real;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2018-05-19 23:15:15 +08:00
|
|
|
use base::allocator::Allocator;
|
|
|
|
use base::dimension::DimMin;
|
|
|
|
use base::storage::Storage;
|
|
|
|
use base::{DefaultAllocator, SquareMatrix};
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
use linalg::LU;
|
2016-12-05 05:44:42 +08:00
|
|
|
|
2017-08-03 01:37:44 +08:00
|
|
|
impl<N: Real, D: DimMin<D, Output = D>, S: Storage<N, D, D>> SquareMatrix<N, D, S> {
|
|
|
|
/// Computes the matrix determinant.
|
|
|
|
///
|
|
|
|
/// If the matrix has a dimension larger than 3, an LU decomposition is used.
|
2016-12-05 05:44:42 +08:00
|
|
|
#[inline]
|
2017-08-03 01:37:44 +08:00
|
|
|
pub fn determinant(&self) -> N
|
2018-10-22 13:00:10 +08:00
|
|
|
where DefaultAllocator: Allocator<N, D, D> + Allocator<(usize, usize), D> {
|
2018-02-02 19:26:35 +08:00
|
|
|
assert!(
|
|
|
|
self.is_square(),
|
|
|
|
"Unable to compute the determinant of a non-square matrix."
|
|
|
|
);
|
2016-12-05 05:44:42 +08:00
|
|
|
let dim = self.shape().0;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
match dim {
|
|
|
|
0 => N::one(),
|
2018-02-02 19:26:35 +08:00
|
|
|
1 => *self.get_unchecked(0, 0),
|
2016-12-05 05:44:42 +08:00
|
|
|
2 => {
|
2018-02-02 19:26:35 +08:00
|
|
|
let m11 = *self.get_unchecked(0, 0);
|
|
|
|
let m12 = *self.get_unchecked(0, 1);
|
|
|
|
let m21 = *self.get_unchecked(1, 0);
|
|
|
|
let m22 = *self.get_unchecked(1, 1);
|
2016-12-05 05:44:42 +08:00
|
|
|
|
|
|
|
m11 * m22 - m21 * m12
|
2018-02-02 19:26:35 +08:00
|
|
|
}
|
2016-12-05 05:44:42 +08:00
|
|
|
3 => {
|
|
|
|
let m11 = *self.get_unchecked(0, 0);
|
|
|
|
let m12 = *self.get_unchecked(0, 1);
|
|
|
|
let m13 = *self.get_unchecked(0, 2);
|
|
|
|
|
|
|
|
let m21 = *self.get_unchecked(1, 0);
|
|
|
|
let m22 = *self.get_unchecked(1, 1);
|
|
|
|
let m23 = *self.get_unchecked(1, 2);
|
|
|
|
|
|
|
|
let m31 = *self.get_unchecked(2, 0);
|
|
|
|
let m32 = *self.get_unchecked(2, 1);
|
|
|
|
let m33 = *self.get_unchecked(2, 2);
|
|
|
|
|
|
|
|
let minor_m12_m23 = m22 * m33 - m32 * m23;
|
|
|
|
let minor_m11_m23 = m21 * m33 - m31 * m23;
|
|
|
|
let minor_m11_m22 = m21 * m32 - m31 * m22;
|
|
|
|
|
|
|
|
m11 * minor_m12_m23 - m12 * minor_m11_m23 + m13 * minor_m11_m22
|
|
|
|
}
|
2018-02-02 19:26:35 +08:00
|
|
|
_ => LU::new(self.clone_owned()).determinant(),
|
2016-12-05 05:44:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|