mirror of
https://gitlab.com/libeigen/eigen.git
synced 2026-04-10 11:34:33 +08:00
165 lines
6.4 KiB
C++
165 lines
6.4 KiB
C++
// This file is part of Eigen, a lightweight C++ template library
|
|
// for linear algebra.
|
|
//
|
|
// Copyright (C) 2016 Rasmus Munk Larsen (rmlarsen@gmail.com)
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla
|
|
// Public License v. 2.0. If a copy of the MPL was not distributed
|
|
// with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
#ifndef EIGEN_CONDITIONESTIMATOR_H
|
|
#define EIGEN_CONDITIONESTIMATOR_H
|
|
|
|
// IWYU pragma: private
|
|
#include "./InternalHeaderCheck.h"
|
|
|
|
namespace Eigen {
|
|
|
|
namespace internal {
|
|
|
|
template <typename Vector, typename RealVector, bool IsComplex>
|
|
struct rcond_compute_sign {
|
|
static inline Vector run(const Vector& v) {
|
|
const RealVector v_abs = v.cwiseAbs();
|
|
return (v_abs.array() == static_cast<typename Vector::RealScalar>(0))
|
|
.select(Vector::Ones(v.size()), v.cwiseQuotient(v_abs));
|
|
}
|
|
};
|
|
|
|
// Partial specialization to avoid elementwise division for real vectors.
|
|
template <typename Vector>
|
|
struct rcond_compute_sign<Vector, Vector, false> {
|
|
static inline Vector run(const Vector& v) {
|
|
return (v.array() < static_cast<typename Vector::RealScalar>(0))
|
|
.select(-Vector::Ones(v.size()), Vector::Ones(v.size()));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* \returns an estimate of ||inv(matrix)||_1 given a decomposition of
|
|
* \a matrix that implements .solve() and .adjoint().solve() methods.
|
|
*
|
|
* This function implements Algorithms 4.1 and 5.1 from
|
|
* Higham, "Experience with a Matrix Norm Estimator",
|
|
* SIAM J. Sci. Stat. Comput., 11(4):804-809, 1990.
|
|
* with Higham's alternating-sign safety-net estimate from
|
|
* Higham and Tisseur, "A Block Algorithm for Matrix 1-Norm Estimation,
|
|
* with an Application to 1-Norm Pseudospectra", SIAM J. Matrix Anal. Appl.,
|
|
* 21(4):1185-1201, 2000.
|
|
*
|
|
* The Hager/Higham gradient ascent uses at most 5 iterations of 2 solves
|
|
* each, giving a total cost of O(n^2).
|
|
*
|
|
* Supports the following decompositions: FullPivLU, PartialPivLU, LDLT, LLT.
|
|
*
|
|
* \sa FullPivLU, PartialPivLU, LDLT, LLT.
|
|
*/
|
|
template <typename Decomposition>
|
|
typename Decomposition::RealScalar rcond_invmatrix_L1_norm_estimate(const Decomposition& dec) {
|
|
typedef typename Decomposition::MatrixType MatrixType;
|
|
typedef typename Decomposition::Scalar Scalar;
|
|
typedef typename Decomposition::RealScalar RealScalar;
|
|
typedef typename internal::plain_col_type<MatrixType>::type Vector;
|
|
typedef typename internal::plain_col_type<MatrixType, RealScalar>::type RealVector;
|
|
const bool is_complex = (NumTraits<Scalar>::IsComplex != 0);
|
|
|
|
eigen_assert(dec.rows() == dec.cols());
|
|
const Index n = dec.rows();
|
|
if (n == 0) return RealScalar(0);
|
|
|
|
// Disable Index to float conversion warning
|
|
#ifdef __INTEL_COMPILER
|
|
#pragma warning push
|
|
#pragma warning(disable : 2259)
|
|
#endif
|
|
Vector v = dec.solve(Vector::Ones(n) / Scalar(n));
|
|
#ifdef __INTEL_COMPILER
|
|
#pragma warning pop
|
|
#endif
|
|
|
|
// lower_bound is a lower bound on
|
|
// ||inv(matrix)||_1 = sup_v ||inv(matrix) v||_1 / ||v||_1
|
|
// and is the objective maximized by the supergradient ascent algorithm below.
|
|
RealScalar lower_bound = v.template lpNorm<1>();
|
|
if (n == 1) return lower_bound;
|
|
|
|
// Gradient ascent: the optimum is achieved at a unit vector e_j. Each
|
|
// iteration follows the supergradient to find which unit vector to probe next.
|
|
RealScalar old_lower_bound = lower_bound;
|
|
Vector sign_vector(n);
|
|
Vector old_sign_vector;
|
|
Index v_max_abs_index = -1;
|
|
Index old_v_max_abs_index = v_max_abs_index;
|
|
for (int k = 0; k < 4; ++k) {
|
|
sign_vector = internal::rcond_compute_sign<Vector, RealVector, is_complex>::run(v);
|
|
if (k > 0 && !is_complex && sign_vector == old_sign_vector) {
|
|
// Break if the sign vector stagnated.
|
|
break;
|
|
}
|
|
// Supergradient: z = A^{-T} * sign(v), pick argmax |z_i|.
|
|
v = dec.adjoint().solve(sign_vector);
|
|
v.real().cwiseAbs().maxCoeff(&v_max_abs_index);
|
|
if (v_max_abs_index == old_v_max_abs_index) {
|
|
// Optimality: supergradient points to the same unit vector.
|
|
break;
|
|
}
|
|
// Probe the best unit vector: v = A^{-1} * e_j.
|
|
v = dec.solve(Vector::Unit(n, v_max_abs_index));
|
|
lower_bound = v.template lpNorm<1>();
|
|
if (lower_bound <= old_lower_bound) {
|
|
// No improvement from the gradient step.
|
|
break;
|
|
}
|
|
if (!is_complex) {
|
|
old_sign_vector = sign_vector;
|
|
}
|
|
old_v_max_abs_index = v_max_abs_index;
|
|
old_lower_bound = lower_bound;
|
|
}
|
|
// Higham's alternating-sign estimate: an independent safety-net that catches
|
|
// cases where the gradient ascent converges to a local maximum due to exact
|
|
// cancellation patterns (especially with permutations and backsubstitutions).
|
|
// v_i = (-1)^i * (1 + i/(n-1)), then estimate = 2*||A^{-1}*v||_1 / (3*n).
|
|
Scalar alternating_sign(RealScalar(1));
|
|
for (Index i = 0; i < n; ++i) {
|
|
// The static_cast is needed when Scalar is complex and RealScalar uses expression templates.
|
|
v[i] = alternating_sign * static_cast<RealScalar>(RealScalar(1) + (RealScalar(i) / (RealScalar(n - 1))));
|
|
alternating_sign = -alternating_sign;
|
|
}
|
|
v = dec.solve(v);
|
|
const RealScalar alt_est = (RealScalar(2) * v.template lpNorm<1>()) / (RealScalar(3) * RealScalar(n));
|
|
return numext::maxi(lower_bound, alt_est);
|
|
}
|
|
|
|
/** \brief Reciprocal condition number estimator.
|
|
*
|
|
* Computing a decomposition of a dense matrix takes O(n^3) operations, while
|
|
* this method estimates the condition number quickly and reliably in O(n^2)
|
|
* operations.
|
|
*
|
|
* \returns an estimate of the reciprocal condition number
|
|
* (1 / (||matrix||_1 * ||inv(matrix)||_1)) of matrix, given ||matrix||_1 and
|
|
* its decomposition. Supports the following decompositions: FullPivLU,
|
|
* PartialPivLU, LDLT, and LLT.
|
|
*
|
|
* \sa FullPivLU, PartialPivLU, LDLT, LLT.
|
|
*/
|
|
template <typename Decomposition>
|
|
typename Decomposition::RealScalar rcond_estimate_helper(typename Decomposition::RealScalar matrix_norm,
|
|
const Decomposition& dec) {
|
|
typedef typename Decomposition::RealScalar RealScalar;
|
|
eigen_assert(dec.rows() == dec.cols());
|
|
if (dec.rows() == 0) return NumTraits<RealScalar>::infinity();
|
|
if (numext::is_exactly_zero(matrix_norm)) return RealScalar(0);
|
|
if (dec.rows() == 1) return RealScalar(1);
|
|
const RealScalar inverse_matrix_norm = rcond_invmatrix_L1_norm_estimate(dec);
|
|
return (numext::is_exactly_zero(inverse_matrix_norm) ? RealScalar(0)
|
|
: (RealScalar(1) / inverse_matrix_norm) / matrix_norm);
|
|
}
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace Eigen
|
|
|
|
#endif
|