symmetric_eigen: allow computing only eigenvalues.

This commit is contained in:
Sébastien Crozet 2017-08-06 17:04:40 +02:00 committed by Sébastien Crozet
parent 6eb0d8a786
commit a7bce9cf3f
2 changed files with 69 additions and 18 deletions

View File

@ -2,8 +2,9 @@ use num_complex::Complex;
use std::ops::MulAssign; use std::ops::MulAssign;
use alga::general::Real; use alga::general::Real;
use core::{MatrixN, VectorN, DefaultAllocator, Matrix2, Vector2}; use core::{MatrixN, VectorN, DefaultAllocator, Matrix2, Vector2, SquareMatrix};
use dimension::{Dim, DimSub, DimDiff, U1, U2}; use dimension::{Dim, DimSub, DimDiff, U1, U2};
use storage::Storage;
use allocator::Allocator; use allocator::Allocator;
use linalg::givens; use linalg::givens;
@ -46,7 +47,19 @@ impl<N: Real, D: Dim> SymmetricEigen<N, D>
/// * `max_niter` maximum total number of iterations performed by the algorithm. If this /// * `max_niter` maximum total number of iterations performed by the algorithm. If this
/// number of iteration is exceeded, `None` is returned. If `niter == 0`, then the algorithm /// number of iteration is exceeded, `None` is returned. If `niter == 0`, then the algorithm
/// continues indefinitely until convergence. /// continues indefinitely until convergence.
pub fn try_new(mut m: MatrixN<N, D>, eps: N, max_niter: usize) -> Option<Self> pub fn try_new(m: MatrixN<N, D>, eps: N, max_niter: usize) -> Option<Self>
where D: DimSub<U1>,
DefaultAllocator: Allocator<N, DimDiff<D, U1>> {
Self::do_decompose(m, true, eps, max_niter).map(|(vals, vecs)| {
SymmetricEigen {
eigenvectors: vecs.unwrap(),
eigenvalues: vals
}
})
}
fn do_decompose(mut m: MatrixN<N, D>, eigenvectors: bool, eps: N, max_niter: usize)
-> Option<(VectorN<N, D>, Option<MatrixN<N, D>>)>
where D: DimSub<U1>, where D: DimSub<U1>,
DefaultAllocator: Allocator<N, DimDiff<D, U1>> { DefaultAllocator: Allocator<N, DimDiff<D, U1>> {
@ -59,15 +72,24 @@ impl<N: Real, D: Dim> SymmetricEigen<N, D>
m /= m_amax; m /= m_amax;
} }
let (mut q, mut diag, mut off_diag) = SymmetricTridiagonal::new(m).unpack(); let (mut q, mut diag, mut off_diag);
if eigenvectors {
let res = SymmetricTridiagonal::new(m).unpack();
q = Some(res.0);
diag = res.1;
off_diag = res.2;
}
else {
let res = SymmetricTridiagonal::new(m).unpack_tridiagonal();
q = None;
diag = res.0;
off_diag = res.1;
}
if dim == 1 { if dim == 1 {
diag *= m_amax; diag *= m_amax;
return Some((diag, q));
return Some(SymmetricEigen {
eigenvectors: q,
eigenvalues: diag
});
} }
let mut niter = 0; let mut niter = 0;
@ -114,8 +136,10 @@ impl<N: Real, D: Dim> SymmetricEigen<N, D>
off_diag[i + 1] *= rot.cos_angle(); off_diag[i + 1] *= rot.cos_angle();
} }
if let Some(ref mut q) = q {
rot.inverse().rotate_rows(&mut q.fixed_columns_mut::<U2>(i)); rot.inverse().rotate_rows(&mut q.fixed_columns_mut::<U2>(i));
} }
}
else { else {
break; break;
} }
@ -134,10 +158,12 @@ impl<N: Real, D: Dim> SymmetricEigen<N, D>
diag[start + 0] = eigvals[0]; diag[start + 0] = eigvals[0];
diag[start + 1] = eigvals[1]; diag[start + 1] = eigvals[1];
if let Some(ref mut q) = q {
if let Some(basis) = basis.try_normalize(eps) { if let Some(basis) = basis.try_normalize(eps) {
let rot = UnitComplex::new_unchecked(Complex::new(basis.x, basis.y)); let rot = UnitComplex::new_unchecked(Complex::new(basis.x, basis.y));
rot.rotate_rows(&mut q.fixed_columns_mut::<U2>(start)); rot.rotate_rows(&mut q.fixed_columns_mut::<U2>(start));
} }
}
end -= 1; end -= 1;
} }
@ -156,12 +182,7 @@ impl<N: Real, D: Dim> SymmetricEigen<N, D>
diag *= m_amax; diag *= m_amax;
// Solve the remaining 2x2 subproblem. Some((diag, q))
Some(SymmetricEigen {
eigenvectors: q,
eigenvalues: diag
})
} }
fn delimit_subproblem(diag: &VectorN<N, D>, fn delimit_subproblem(diag: &VectorN<N, D>,
@ -236,6 +257,28 @@ pub fn wilkinson_shift<N: Real>(tmm: N, tnn: N, tmn: N) -> N {
} }
} }
/*
*
* Computations of eigenvalues for symmetric matrices.
*
*/
impl<N: Real, D: DimSub<U1>, S: Storage<N, D, D>> SquareMatrix<N, D, S>
where DefaultAllocator: Allocator<N, D, D> +
Allocator<N, D> +
Allocator<N, DimDiff<D, U1>> {
/// Computes the eigenvalues of this symmetric matrix.
///
/// Only the lower-triangular part of the matrix is read.
pub fn symmetric_eigenvalues(&self) -> VectorN<N, D> {
SymmetricEigen::do_decompose(self.clone_owned(), false, N::default_epsilon(), 0).unwrap().0
}
}
#[cfg(test)] #[cfg(test)]
mod test { mod test {
use core::Matrix2; use core::Matrix2;

View File

@ -71,6 +71,14 @@ impl<N: Real, D: DimSub<U1>> SymmetricTridiagonal<N, D>
(q, diag, self.off_diagonal) (q, diag, self.off_diagonal)
} }
/// Retrieve the diagonal, and off diagonal elements of this decomposition.
pub fn unpack_tridiagonal(self) -> (VectorN<N, D>, VectorN<N, DimDiff<D, U1>>)
where DefaultAllocator: Allocator<N, D> {
let diag = self.diagonal();
(diag, self.off_diagonal)
}
/// The diagonal components of this decomposition. /// The diagonal components of this decomposition.
pub fn diagonal(&self) -> VectorN<N, D> pub fn diagonal(&self) -> VectorN<N, D>
where DefaultAllocator: Allocator<N, D> { where DefaultAllocator: Allocator<N, D> {