2021-04-12 18:14:16 +08:00
|
|
|
#[cfg(feature = "serde-serialize-no-std")]
|
2020-03-21 19:16:46 +08:00
|
|
|
use serde::{Deserialize, Serialize};
|
2017-08-14 01:53:04 +08:00
|
|
|
|
2018-05-19 23:15:15 +08:00
|
|
|
use num_complex::Complex;
|
2020-03-21 19:16:46 +08:00
|
|
|
use simba::scalar::ComplexField;
|
2018-05-19 23:15:15 +08:00
|
|
|
use std::cmp;
|
|
|
|
use std::fmt::Display;
|
2017-08-03 01:37:44 +08:00
|
|
|
use std::ops::MulAssign;
|
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::allocator::Allocator;
|
|
|
|
use crate::base::dimension::{Dim, DimDiff, DimSub, Dynamic, U1, U2, U3};
|
|
|
|
use crate::base::storage::Storage;
|
2020-03-21 19:16:46 +08:00
|
|
|
use crate::base::{
|
2021-04-11 17:00:38 +08:00
|
|
|
DefaultAllocator, Hessenberg, OMatrix, OVector, SquareMatrix, Unit, Vector2, Vector3,
|
2020-03-21 19:16:46 +08:00
|
|
|
};
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::constraint::{DimEq, ShapeConstraint};
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2019-03-23 21:29:07 +08:00
|
|
|
use crate::geometry::{Reflection, UnitComplex};
|
|
|
|
use crate::linalg::householder;
|
|
|
|
use crate::linalg::Schur;
|
2017-08-03 01:37:44 +08:00
|
|
|
|
2019-03-03 02:33:49 +08:00
|
|
|
/// Eigendecomposition of a real matrix with real eigenvalues (or complex eigen values for complex matrices).
|
2021-04-12 18:14:16 +08:00
|
|
|
#[cfg_attr(feature = "serde-serialize-no-std", derive(Serialize, Deserialize))]
|
2018-05-19 23:15:15 +08:00
|
|
|
#[cfg_attr(
|
2021-04-12 18:14:16 +08:00
|
|
|
feature = "serde-serialize-no-std",
|
2021-04-11 17:00:38 +08:00
|
|
|
serde(bound(serialize = "DefaultAllocator: Allocator<T, D>,
|
|
|
|
OVector<T, D>: Serialize,
|
|
|
|
OMatrix<T, D, D>: Serialize"))
|
2018-05-19 23:15:15 +08:00
|
|
|
)]
|
|
|
|
#[cfg_attr(
|
2021-04-12 18:14:16 +08:00
|
|
|
feature = "serde-serialize-no-std",
|
2021-04-11 17:00:38 +08:00
|
|
|
serde(bound(deserialize = "DefaultAllocator: Allocator<T, D>,
|
|
|
|
OVector<T, D>: Serialize,
|
|
|
|
OMatrix<T, D, D>: Deserialize<'de>"))
|
2018-05-19 23:15:15 +08:00
|
|
|
)]
|
2017-08-14 01:53:00 +08:00
|
|
|
#[derive(Clone, Debug)]
|
2021-04-11 17:00:38 +08:00
|
|
|
pub struct Eigen<T: ComplexField, D: Dim>
|
|
|
|
where
|
|
|
|
DefaultAllocator: Allocator<T, D, D> + Allocator<T, D>,
|
2018-05-19 23:15:15 +08:00
|
|
|
{
|
2021-04-11 17:00:38 +08:00
|
|
|
pub eigenvectors: OMatrix<T, D, D>,
|
|
|
|
pub eigenvalues: OVector<T, D>,
|
2017-08-03 01:37:44 +08:00
|
|
|
}
|
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: ComplexField, D: Dim> Copy for Eigen<T, D>
|
2018-05-19 23:15:15 +08:00
|
|
|
where
|
2021-04-11 17:00:38 +08:00
|
|
|
DefaultAllocator: Allocator<T, D, D> + Allocator<T, D>,
|
|
|
|
OMatrix<T, D, D>: Copy,
|
|
|
|
OVector<T, D>: Copy,
|
2018-05-19 23:15:15 +08:00
|
|
|
{
|
|
|
|
}
|
2017-08-14 01:53:00 +08:00
|
|
|
|
2021-04-11 17:00:38 +08:00
|
|
|
impl<T: ComplexField, D: Dim> Eigen<T, D>
|
2018-05-19 23:15:15 +08:00
|
|
|
where
|
|
|
|
D: DimSub<U1>, // For Hessenberg.
|
|
|
|
ShapeConstraint: DimEq<Dynamic, DimDiff<D, U1>>, // For Hessenberg.
|
2021-04-11 17:00:38 +08:00
|
|
|
DefaultAllocator: Allocator<T, D, DimDiff<D, U1>>
|
|
|
|
+ Allocator<T, DimDiff<D, U1>>
|
|
|
|
+ Allocator<T, D, D>
|
|
|
|
+ Allocator<T, D>,
|
2018-05-19 23:15:15 +08:00
|
|
|
// XXX: for debug
|
|
|
|
DefaultAllocator: Allocator<usize, D, D>,
|
2021-04-11 17:00:38 +08:00
|
|
|
OMatrix<T, D, D>: Display,
|
2018-05-19 23:15:15 +08:00
|
|
|
{
|
2019-03-03 02:33:49 +08:00
|
|
|
/// Computes the eigendecomposition of a diagonalizable matrix with Complex eigenvalues.
|
2021-04-11 17:00:38 +08:00
|
|
|
pub fn new(m: OMatrix<T, D, D>) -> Option<Eigen<T, D>> {
|
2018-05-19 23:15:15 +08:00
|
|
|
assert!(
|
|
|
|
m.is_square(),
|
|
|
|
"Unable to compute the eigendecomposition of a non-square matrix."
|
|
|
|
);
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
let dim = m.nrows();
|
2019-03-23 18:46:56 +08:00
|
|
|
let (mut eigenvectors, mut eigenvalues) = Schur::new(m, 0).unwrap().unpack();
|
2017-08-03 01:37:44 +08:00
|
|
|
|
|
|
|
println!("Schur eigenvalues: {}", eigenvalues);
|
|
|
|
|
2019-03-03 02:33:49 +08:00
|
|
|
// Check that the eigenvalues are all Complex.
|
2018-05-19 23:15:15 +08:00
|
|
|
for i in 0..dim - 1 {
|
2017-08-03 01:37:44 +08:00
|
|
|
if !eigenvalues[(i + 1, i)].is_zero() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-19 23:15:15 +08:00
|
|
|
for j in 1..dim {
|
|
|
|
for i in 0..j {
|
2017-08-03 01:37:44 +08:00
|
|
|
let diff = eigenvalues[(i, i)] - eigenvalues[(j, j)];
|
|
|
|
|
|
|
|
if diff.is_zero() && !eigenvalues[(i, j)].is_zero() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let z = -eigenvalues[(i, j)] / diff;
|
|
|
|
|
2018-05-19 23:15:15 +08:00
|
|
|
for k in j + 1..dim {
|
2017-08-03 01:37:44 +08:00
|
|
|
eigenvalues[(i, k)] -= z * eigenvalues[(j, k)];
|
|
|
|
}
|
|
|
|
|
2018-05-19 23:15:15 +08:00
|
|
|
for k in 0..dim {
|
2017-08-03 01:37:44 +08:00
|
|
|
eigenvectors[(k, j)] += z * eigenvectors[(k, i)];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normalize the eigenvector basis.
|
2018-05-19 23:15:15 +08:00
|
|
|
for i in 0..dim {
|
2017-08-03 01:37:44 +08:00
|
|
|
let _ = eigenvectors.column_mut(i).normalize_mut();
|
|
|
|
}
|
|
|
|
|
2019-03-03 02:33:49 +08:00
|
|
|
Some(Eigen {
|
|
|
|
eigenvectors,
|
2018-05-19 23:15:15 +08:00
|
|
|
eigenvalues: eigenvalues.diagonal(),
|
2017-08-03 01:37:44 +08:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|