mirror of
https://gitlab.com/libeigen/eigen.git
synced 2025-09-11 00:43:13 +08:00
Allow user to compute only the eigenvalues and not the eigenvectors.
This commit is contained in:
parent
609941380a
commit
8dc947821b
@ -56,7 +56,10 @@
|
|||||||
template<typename _MatrixType> class ComplexEigenSolver
|
template<typename _MatrixType> class ComplexEigenSolver
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
/** \brief Synonym for the template parameter \p _MatrixType. */
|
||||||
typedef _MatrixType MatrixType;
|
typedef _MatrixType MatrixType;
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
RowsAtCompileTime = MatrixType::RowsAtCompileTime,
|
RowsAtCompileTime = MatrixType::RowsAtCompileTime,
|
||||||
ColsAtCompileTime = MatrixType::ColsAtCompileTime,
|
ColsAtCompileTime = MatrixType::ColsAtCompileTime,
|
||||||
@ -65,12 +68,12 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
MaxColsAtCompileTime = MatrixType::MaxColsAtCompileTime
|
MaxColsAtCompileTime = MatrixType::MaxColsAtCompileTime
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \brief Scalar type for matrices of type \p _MatrixType. */
|
/** \brief Scalar type for matrices of type #MatrixType. */
|
||||||
typedef typename MatrixType::Scalar Scalar;
|
typedef typename MatrixType::Scalar Scalar;
|
||||||
typedef typename NumTraits<Scalar>::Real RealScalar;
|
typedef typename NumTraits<Scalar>::Real RealScalar;
|
||||||
typedef typename MatrixType::Index Index;
|
typedef typename MatrixType::Index Index;
|
||||||
|
|
||||||
/** \brief Complex scalar type for \p _MatrixType.
|
/** \brief Complex scalar type for #MatrixType.
|
||||||
*
|
*
|
||||||
* This is \c std::complex<Scalar> if #Scalar is real (e.g.,
|
* This is \c std::complex<Scalar> if #Scalar is real (e.g.,
|
||||||
* \c float or \c double) and just \c Scalar if #Scalar is
|
* \c float or \c double) and just \c Scalar if #Scalar is
|
||||||
@ -81,14 +84,14 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
/** \brief Type for vector of eigenvalues as returned by eigenvalues().
|
/** \brief Type for vector of eigenvalues as returned by eigenvalues().
|
||||||
*
|
*
|
||||||
* This is a column vector with entries of type #ComplexScalar.
|
* This is a column vector with entries of type #ComplexScalar.
|
||||||
* The length of the vector is the size of \p _MatrixType.
|
* The length of the vector is the size of #MatrixType.
|
||||||
*/
|
*/
|
||||||
typedef Matrix<ComplexScalar, ColsAtCompileTime, 1, Options, MaxColsAtCompileTime, 1> EigenvalueType;
|
typedef Matrix<ComplexScalar, ColsAtCompileTime, 1, Options, MaxColsAtCompileTime, 1> EigenvalueType;
|
||||||
|
|
||||||
/** \brief Type for matrix of eigenvectors as returned by eigenvectors().
|
/** \brief Type for matrix of eigenvectors as returned by eigenvectors().
|
||||||
*
|
*
|
||||||
* This is a square matrix with entries of type #ComplexScalar.
|
* This is a square matrix with entries of type #ComplexScalar.
|
||||||
* The size is the same as the size of \p _MatrixType.
|
* The size is the same as the size of #MatrixType.
|
||||||
*/
|
*/
|
||||||
typedef Matrix<ComplexScalar, RowsAtCompileTime, ColsAtCompileTime, Options, MaxRowsAtCompileTime, ColsAtCompileTime> EigenvectorType;
|
typedef Matrix<ComplexScalar, RowsAtCompileTime, ColsAtCompileTime, Options, MaxRowsAtCompileTime, ColsAtCompileTime> EigenvectorType;
|
||||||
|
|
||||||
@ -102,6 +105,7 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
m_eivalues(),
|
m_eivalues(),
|
||||||
m_schur(),
|
m_schur(),
|
||||||
m_isInitialized(false),
|
m_isInitialized(false),
|
||||||
|
m_eigenvectorsOk(false),
|
||||||
m_matX()
|
m_matX()
|
||||||
{}
|
{}
|
||||||
|
|
||||||
@ -116,40 +120,46 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
m_eivalues(size),
|
m_eivalues(size),
|
||||||
m_schur(size),
|
m_schur(size),
|
||||||
m_isInitialized(false),
|
m_isInitialized(false),
|
||||||
|
m_eigenvectorsOk(false),
|
||||||
m_matX(size, size)
|
m_matX(size, size)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
/** \brief Constructor; computes eigendecomposition of given matrix.
|
/** \brief Constructor; computes eigendecomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
||||||
|
* \param[in] computeEigenvectors If true, both the eigenvectors and the
|
||||||
|
* eigenvalues are computed; if false, only the eigenvalues are
|
||||||
|
* computed.
|
||||||
*
|
*
|
||||||
* This constructor calls compute() to compute the eigendecomposition.
|
* This constructor calls compute() to compute the eigendecomposition.
|
||||||
*/
|
*/
|
||||||
ComplexEigenSolver(const MatrixType& matrix)
|
ComplexEigenSolver(const MatrixType& matrix, bool computeEigenvectors = true)
|
||||||
: m_eivec(matrix.rows(),matrix.cols()),
|
: m_eivec(matrix.rows(),matrix.cols()),
|
||||||
m_eivalues(matrix.cols()),
|
m_eivalues(matrix.cols()),
|
||||||
m_schur(matrix.rows()),
|
m_schur(matrix.rows()),
|
||||||
m_isInitialized(false),
|
m_isInitialized(false),
|
||||||
|
m_eigenvectorsOk(false),
|
||||||
m_matX(matrix.rows(),matrix.cols())
|
m_matX(matrix.rows(),matrix.cols())
|
||||||
{
|
{
|
||||||
compute(matrix);
|
compute(matrix, computeEigenvectors);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Returns the eigenvectors of given matrix.
|
/** \brief Returns the eigenvectors of given matrix.
|
||||||
*
|
*
|
||||||
* \returns A const reference to the matrix whose columns are the eigenvectors.
|
* \returns A const reference to the matrix whose columns are the eigenvectors.
|
||||||
*
|
*
|
||||||
* It is assumed that either the constructor
|
* \pre Either the constructor
|
||||||
* ComplexEigenSolver(const MatrixType& matrix) or the member
|
* ComplexEigenSolver(const MatrixType& matrix, bool) or the member
|
||||||
* function compute(const MatrixType& matrix) has been called
|
* function compute(const MatrixType& matrix, bool) has been called before
|
||||||
* before to compute the eigendecomposition of a matrix. This
|
* to compute the eigendecomposition of a matrix, and
|
||||||
* function returns a matrix whose columns are the
|
* \p computeEigenvectors was set to true (the default).
|
||||||
* eigenvectors. Column \f$ k \f$ is an eigenvector
|
*
|
||||||
* corresponding to eigenvalue number \f$ k \f$ as returned by
|
* This function returns a matrix whose columns are the eigenvectors. Column
|
||||||
* eigenvalues(). The eigenvectors are normalized to have
|
* \f$ k \f$ is an eigenvector corresponding to eigenvalue number \f$ k
|
||||||
* (Euclidean) norm equal to one. The matrix returned by this
|
* \f$ as returned by eigenvalues(). The eigenvectors are normalized to
|
||||||
* function is the matrix \f$ V \f$ in the eigendecomposition \f$
|
* have (Euclidean) norm equal to one. The matrix returned by this
|
||||||
* A = V D V^{-1} \f$, if it exists.
|
* function is the matrix \f$ V \f$ in the eigendecomposition \f$ A = V D
|
||||||
|
* V^{-1} \f$, if it exists.
|
||||||
*
|
*
|
||||||
* Example: \include ComplexEigenSolver_eigenvectors.cpp
|
* Example: \include ComplexEigenSolver_eigenvectors.cpp
|
||||||
* Output: \verbinclude ComplexEigenSolver_eigenvectors.out
|
* Output: \verbinclude ComplexEigenSolver_eigenvectors.out
|
||||||
@ -157,6 +167,7 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
const EigenvectorType& eigenvectors() const
|
const EigenvectorType& eigenvectors() const
|
||||||
{
|
{
|
||||||
ei_assert(m_isInitialized && "ComplexEigenSolver is not initialized.");
|
ei_assert(m_isInitialized && "ComplexEigenSolver is not initialized.");
|
||||||
|
ei_assert(m_eigenvectorsOk && "The eigenvectors have not been computed together with the eigenvalues.");
|
||||||
return m_eivec;
|
return m_eivec;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,11 +175,12 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
*
|
*
|
||||||
* \returns A const reference to the column vector containing the eigenvalues.
|
* \returns A const reference to the column vector containing the eigenvalues.
|
||||||
*
|
*
|
||||||
* It is assumed that either the constructor
|
* \pre Either the constructor
|
||||||
* ComplexEigenSolver(const MatrixType& matrix) or the member
|
* ComplexEigenSolver(const MatrixType& matrix, bool) or the member
|
||||||
* function compute(const MatrixType& matrix) has been called
|
* function compute(const MatrixType& matrix, bool) has been called before
|
||||||
* before to compute the eigendecomposition of a matrix. This
|
* to compute the eigendecomposition of a matrix.
|
||||||
* function returns a column vector containing the
|
*
|
||||||
|
* This function returns a column vector containing the
|
||||||
* eigenvalues. Eigenvalues are repeated according to their
|
* eigenvalues. Eigenvalues are repeated according to their
|
||||||
* algebraic multiplicity, so there are as many eigenvalues as
|
* algebraic multiplicity, so there are as many eigenvalues as
|
||||||
* rows in the matrix.
|
* rows in the matrix.
|
||||||
@ -185,10 +197,14 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
/** \brief Computes eigendecomposition of given matrix.
|
/** \brief Computes eigendecomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
||||||
|
* \param[in] computeEigenvectors If true, both the eigenvectors and the
|
||||||
|
* eigenvalues are computed; if false, only the eigenvalues are
|
||||||
|
* computed.
|
||||||
*
|
*
|
||||||
* This function computes the eigenvalues and eigenvectors of \p
|
* This function computes the eigenvalues of the complex matrix \p matrix.
|
||||||
* matrix. The eigenvalues() and eigenvectors() functions can be
|
* The eigenvalues() function can be used to retrieve them. If
|
||||||
* used to retrieve the computed eigendecomposition.
|
* \p computeEigenvectors is true, then the eigenvectors are also computed
|
||||||
|
* and can be retrieved by calling eigenvectors().
|
||||||
*
|
*
|
||||||
* The matrix is first reduced to Schur form using the
|
* The matrix is first reduced to Schur form using the
|
||||||
* ComplexSchur class. The Schur decomposition is then used to
|
* ComplexSchur class. The Schur decomposition is then used to
|
||||||
@ -201,19 +217,20 @@ template<typename _MatrixType> class ComplexEigenSolver
|
|||||||
* Example: \include ComplexEigenSolver_compute.cpp
|
* Example: \include ComplexEigenSolver_compute.cpp
|
||||||
* Output: \verbinclude ComplexEigenSolver_compute.out
|
* Output: \verbinclude ComplexEigenSolver_compute.out
|
||||||
*/
|
*/
|
||||||
void compute(const MatrixType& matrix);
|
void compute(const MatrixType& matrix, bool computeEigenvectors = true);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
EigenvectorType m_eivec;
|
EigenvectorType m_eivec;
|
||||||
EigenvalueType m_eivalues;
|
EigenvalueType m_eivalues;
|
||||||
ComplexSchur<MatrixType> m_schur;
|
ComplexSchur<MatrixType> m_schur;
|
||||||
bool m_isInitialized;
|
bool m_isInitialized;
|
||||||
|
bool m_eigenvectorsOk;
|
||||||
EigenvectorType m_matX;
|
EigenvectorType m_matX;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
void ComplexEigenSolver<MatrixType>::compute(const MatrixType& matrix)
|
void ComplexEigenSolver<MatrixType>::compute(const MatrixType& matrix, bool computeEigenvectors)
|
||||||
{
|
{
|
||||||
// this code is inspired from Jampack
|
// this code is inspired from Jampack
|
||||||
assert(matrix.cols() == matrix.rows());
|
assert(matrix.cols() == matrix.rows());
|
||||||
@ -222,9 +239,11 @@ void ComplexEigenSolver<MatrixType>::compute(const MatrixType& matrix)
|
|||||||
|
|
||||||
// Step 1: Do a complex Schur decomposition, A = U T U^*
|
// Step 1: Do a complex Schur decomposition, A = U T U^*
|
||||||
// The eigenvalues are on the diagonal of T.
|
// The eigenvalues are on the diagonal of T.
|
||||||
m_schur.compute(matrix);
|
m_schur.compute(matrix, computeEigenvectors);
|
||||||
m_eivalues = m_schur.matrixT().diagonal();
|
m_eivalues = m_schur.matrixT().diagonal();
|
||||||
|
|
||||||
|
if(computeEigenvectors)
|
||||||
|
{
|
||||||
// Step 2: Compute X such that T = X D X^(-1), where D is the diagonal of T.
|
// Step 2: Compute X such that T = X D X^(-1), where D is the diagonal of T.
|
||||||
// The matrix X is unit triangular.
|
// The matrix X is unit triangular.
|
||||||
m_matX = EigenvectorType::Zero(n, n);
|
m_matX = EigenvectorType::Zero(n, n);
|
||||||
@ -255,7 +274,10 @@ void ComplexEigenSolver<MatrixType>::compute(const MatrixType& matrix)
|
|||||||
{
|
{
|
||||||
m_eivec.col(k).normalize();
|
m_eivec.col(k).normalize();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
m_isInitialized = true;
|
m_isInitialized = true;
|
||||||
|
m_eigenvectorsOk = computeEigenvectors;
|
||||||
|
|
||||||
// Step 4: Sort the eigenvalues
|
// Step 4: Sort the eigenvalues
|
||||||
for (Index i=0; i<n; i++)
|
for (Index i=0; i<n; i++)
|
||||||
@ -266,6 +288,7 @@ void ComplexEigenSolver<MatrixType>::compute(const MatrixType& matrix)
|
|||||||
{
|
{
|
||||||
k += i;
|
k += i;
|
||||||
std::swap(m_eivalues[k],m_eivalues[i]);
|
std::swap(m_eivalues[k],m_eivalues[i]);
|
||||||
|
if(computeEigenvectors)
|
||||||
m_eivec.col(i).swap(m_eivec.col(k));
|
m_eivec.col(i).swap(m_eivec.col(k));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -58,15 +58,15 @@
|
|||||||
*
|
*
|
||||||
* Call the function compute() to compute the eigenvalues and eigenvectors of
|
* Call the function compute() to compute the eigenvalues and eigenvectors of
|
||||||
* a given matrix. Alternatively, you can use the
|
* a given matrix. Alternatively, you can use the
|
||||||
* EigenSolver(const MatrixType&) constructor which computes the eigenvalues
|
* EigenSolver(const MatrixType&, bool) constructor which computes the
|
||||||
* and eigenvectors at construction time. Once the eigenvalue and eigenvectors
|
* eigenvalues and eigenvectors at construction time. Once the eigenvalue and
|
||||||
* are computed, they can be retrieved with the eigenvalues() and
|
* eigenvectors are computed, they can be retrieved with the eigenvalues() and
|
||||||
* eigenvectors() functions. The pseudoEigenvalueMatrix() and
|
* eigenvectors() functions. The pseudoEigenvalueMatrix() and
|
||||||
* pseudoEigenvectors() methods allow the construction of the
|
* pseudoEigenvectors() methods allow the construction of the
|
||||||
* pseudo-eigendecomposition.
|
* pseudo-eigendecomposition.
|
||||||
*
|
*
|
||||||
* The documentation for EigenSolver(const MatrixType&) contains an example of
|
* The documentation for EigenSolver(const MatrixType&, bool) contains an
|
||||||
* the typical use of this class.
|
* example of the typical use of this class.
|
||||||
*
|
*
|
||||||
* \note The implementation is adapted from
|
* \note The implementation is adapted from
|
||||||
* <a href="http://math.nist.gov/javanumerics/jama/">JAMA</a> (public domain).
|
* <a href="http://math.nist.gov/javanumerics/jama/">JAMA</a> (public domain).
|
||||||
@ -78,7 +78,9 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
/** \brief Synonym for the template parameter \p _MatrixType. */
|
||||||
typedef _MatrixType MatrixType;
|
typedef _MatrixType MatrixType;
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
RowsAtCompileTime = MatrixType::RowsAtCompileTime,
|
RowsAtCompileTime = MatrixType::RowsAtCompileTime,
|
||||||
ColsAtCompileTime = MatrixType::ColsAtCompileTime,
|
ColsAtCompileTime = MatrixType::ColsAtCompileTime,
|
||||||
@ -87,12 +89,12 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
MaxColsAtCompileTime = MatrixType::MaxColsAtCompileTime
|
MaxColsAtCompileTime = MatrixType::MaxColsAtCompileTime
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \brief Scalar type for matrices of type \p _MatrixType. */
|
/** \brief Scalar type for matrices of type #MatrixType. */
|
||||||
typedef typename MatrixType::Scalar Scalar;
|
typedef typename MatrixType::Scalar Scalar;
|
||||||
typedef typename NumTraits<Scalar>::Real RealScalar;
|
typedef typename NumTraits<Scalar>::Real RealScalar;
|
||||||
typedef typename MatrixType::Index Index;
|
typedef typename MatrixType::Index Index;
|
||||||
|
|
||||||
/** \brief Complex scalar type for \p _MatrixType.
|
/** \brief Complex scalar type for #MatrixType.
|
||||||
*
|
*
|
||||||
* This is \c std::complex<Scalar> if #Scalar is real (e.g.,
|
* This is \c std::complex<Scalar> if #Scalar is real (e.g.,
|
||||||
* \c float or \c double) and just \c Scalar if #Scalar is
|
* \c float or \c double) and just \c Scalar if #Scalar is
|
||||||
@ -103,27 +105,27 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
/** \brief Type for vector of eigenvalues as returned by eigenvalues().
|
/** \brief Type for vector of eigenvalues as returned by eigenvalues().
|
||||||
*
|
*
|
||||||
* This is a column vector with entries of type #ComplexScalar.
|
* This is a column vector with entries of type #ComplexScalar.
|
||||||
* The length of the vector is the size of \p _MatrixType.
|
* The length of the vector is the size of #MatrixType.
|
||||||
*/
|
*/
|
||||||
typedef Matrix<ComplexScalar, ColsAtCompileTime, 1, Options & ~RowMajor, MaxColsAtCompileTime, 1> EigenvalueType;
|
typedef Matrix<ComplexScalar, ColsAtCompileTime, 1, Options & ~RowMajor, MaxColsAtCompileTime, 1> EigenvalueType;
|
||||||
|
|
||||||
/** \brief Type for matrix of eigenvectors as returned by eigenvectors().
|
/** \brief Type for matrix of eigenvectors as returned by eigenvectors().
|
||||||
*
|
*
|
||||||
* This is a square matrix with entries of type #ComplexScalar.
|
* This is a square matrix with entries of type #ComplexScalar.
|
||||||
* The size is the same as the size of \p _MatrixType.
|
* The size is the same as the size of #MatrixType.
|
||||||
*/
|
*/
|
||||||
typedef Matrix<ComplexScalar, RowsAtCompileTime, ColsAtCompileTime, Options, MaxRowsAtCompileTime, MaxColsAtCompileTime> EigenvectorsType;
|
typedef Matrix<ComplexScalar, RowsAtCompileTime, ColsAtCompileTime, Options, MaxRowsAtCompileTime, MaxColsAtCompileTime> EigenvectorsType;
|
||||||
|
|
||||||
/** \brief Default constructor.
|
/** \brief Default constructor.
|
||||||
*
|
*
|
||||||
* The default constructor is useful in cases in which the user intends to
|
* The default constructor is useful in cases in which the user intends to
|
||||||
* perform decompositions via EigenSolver::compute(const MatrixType&).
|
* perform decompositions via EigenSolver::compute(const MatrixType&, bool).
|
||||||
*
|
*
|
||||||
* \sa compute() for an example.
|
* \sa compute() for an example.
|
||||||
*/
|
*/
|
||||||
EigenSolver() : m_eivec(), m_eivalues(), m_isInitialized(false), m_realSchur(), m_matT(), m_tmp() {}
|
EigenSolver() : m_eivec(), m_eivalues(), m_isInitialized(false), m_realSchur(), m_matT(), m_tmp() {}
|
||||||
|
|
||||||
/** \brief Default Constructor with memory preallocation
|
/** \brief Default constructor with memory preallocation
|
||||||
*
|
*
|
||||||
* Like the default constructor but with preallocation of the internal data
|
* Like the default constructor but with preallocation of the internal data
|
||||||
* according to the specified problem \a size.
|
* according to the specified problem \a size.
|
||||||
@ -133,6 +135,7 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
: m_eivec(size, size),
|
: m_eivec(size, size),
|
||||||
m_eivalues(size),
|
m_eivalues(size),
|
||||||
m_isInitialized(false),
|
m_isInitialized(false),
|
||||||
|
m_eigenvectorsOk(false),
|
||||||
m_realSchur(size),
|
m_realSchur(size),
|
||||||
m_matT(size, size),
|
m_matT(size, size),
|
||||||
m_tmp(size)
|
m_tmp(size)
|
||||||
@ -141,6 +144,9 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
/** \brief Constructor; computes eigendecomposition of given matrix.
|
/** \brief Constructor; computes eigendecomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
||||||
|
* \param[in] computeEigenvectors If true, both the eigenvectors and the
|
||||||
|
* eigenvalues are computed; if false, only the eigenvalues are
|
||||||
|
* computed.
|
||||||
*
|
*
|
||||||
* This constructor calls compute() to compute the eigenvalues
|
* This constructor calls compute() to compute the eigenvalues
|
||||||
* and eigenvectors.
|
* and eigenvectors.
|
||||||
@ -150,23 +156,26 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
*
|
*
|
||||||
* \sa compute()
|
* \sa compute()
|
||||||
*/
|
*/
|
||||||
EigenSolver(const MatrixType& matrix)
|
EigenSolver(const MatrixType& matrix, bool computeEigenvectors = true)
|
||||||
: m_eivec(matrix.rows(), matrix.cols()),
|
: m_eivec(matrix.rows(), matrix.cols()),
|
||||||
m_eivalues(matrix.cols()),
|
m_eivalues(matrix.cols()),
|
||||||
m_isInitialized(false),
|
m_isInitialized(false),
|
||||||
|
m_eigenvectorsOk(false),
|
||||||
m_realSchur(matrix.cols()),
|
m_realSchur(matrix.cols()),
|
||||||
m_matT(matrix.rows(), matrix.cols()),
|
m_matT(matrix.rows(), matrix.cols()),
|
||||||
m_tmp(matrix.cols())
|
m_tmp(matrix.cols())
|
||||||
{
|
{
|
||||||
compute(matrix);
|
compute(matrix, computeEigenvectors);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Returns the eigenvectors of given matrix.
|
/** \brief Returns the eigenvectors of given matrix.
|
||||||
*
|
*
|
||||||
* \returns %Matrix whose columns are the (possibly complex) eigenvectors.
|
* \returns %Matrix whose columns are the (possibly complex) eigenvectors.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor EigenSolver(const MatrixType&) or the
|
* \pre Either the constructor
|
||||||
* member function compute(const MatrixType&) has been called before.
|
* EigenSolver(const MatrixType&,bool) or the member function
|
||||||
|
* compute(const MatrixType&, bool) has been called before, and
|
||||||
|
* \p computeEigenvectors was set to true (the default).
|
||||||
*
|
*
|
||||||
* Column \f$ k \f$ of the returned matrix is an eigenvector corresponding
|
* Column \f$ k \f$ of the returned matrix is an eigenvector corresponding
|
||||||
* to eigenvalue number \f$ k \f$ as returned by eigenvalues(). The
|
* to eigenvalue number \f$ k \f$ as returned by eigenvalues(). The
|
||||||
@ -185,9 +194,10 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
*
|
*
|
||||||
* \returns Const reference to matrix whose columns are the pseudo-eigenvectors.
|
* \returns Const reference to matrix whose columns are the pseudo-eigenvectors.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor EigenSolver(const MatrixType&) or
|
* \pre Either the constructor
|
||||||
* the member function compute(const MatrixType&) has been called
|
* EigenSolver(const MatrixType&,bool) or the member function
|
||||||
* before.
|
* compute(const MatrixType&, bool) has been called before, and
|
||||||
|
* \p computeEigenvectors was set to true (the default).
|
||||||
*
|
*
|
||||||
* The real matrix \f$ V \f$ returned by this function and the
|
* The real matrix \f$ V \f$ returned by this function and the
|
||||||
* block-diagonal matrix \f$ D \f$ returned by pseudoEigenvalueMatrix()
|
* block-diagonal matrix \f$ D \f$ returned by pseudoEigenvalueMatrix()
|
||||||
@ -201,6 +211,7 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
const MatrixType& pseudoEigenvectors() const
|
const MatrixType& pseudoEigenvectors() const
|
||||||
{
|
{
|
||||||
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
||||||
|
ei_assert(m_eigenvectorsOk && "The eigenvectors have not been computed together with the eigenvalues.");
|
||||||
return m_eivec;
|
return m_eivec;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -208,8 +219,9 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
*
|
*
|
||||||
* \returns A block-diagonal matrix.
|
* \returns A block-diagonal matrix.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor EigenSolver(const MatrixType&) or the
|
* \pre Either the constructor
|
||||||
* member function compute(const MatrixType&) has been called before.
|
* EigenSolver(const MatrixType&,bool) or the member function
|
||||||
|
* compute(const MatrixType&, bool) has been called before.
|
||||||
*
|
*
|
||||||
* The matrix \f$ D \f$ returned by this function is real and
|
* The matrix \f$ D \f$ returned by this function is real and
|
||||||
* block-diagonal. The blocks on the diagonal are either 1-by-1 or 2-by-2
|
* block-diagonal. The blocks on the diagonal are either 1-by-1 or 2-by-2
|
||||||
@ -226,8 +238,9 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
*
|
*
|
||||||
* \returns A const reference to the column vector containing the eigenvalues.
|
* \returns A const reference to the column vector containing the eigenvalues.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor EigenSolver(const MatrixType&) or the
|
* \pre Either the constructor
|
||||||
* member function compute(const MatrixType&) has been called before.
|
* EigenSolver(const MatrixType&,bool) or the member function
|
||||||
|
* compute(const MatrixType&, bool) has been called before.
|
||||||
*
|
*
|
||||||
* The eigenvalues are repeated according to their algebraic multiplicity,
|
* The eigenvalues are repeated according to their algebraic multiplicity,
|
||||||
* so there are as many eigenvalues as rows in the matrix.
|
* so there are as many eigenvalues as rows in the matrix.
|
||||||
@ -247,34 +260,40 @@ template<typename _MatrixType> class EigenSolver
|
|||||||
/** \brief Computes eigendecomposition of given matrix.
|
/** \brief Computes eigendecomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
* \param[in] matrix Square matrix whose eigendecomposition is to be computed.
|
||||||
|
* \param[in] computeEigenvectors If true, both the eigenvectors and the
|
||||||
|
* eigenvalues are computed; if false, only the eigenvalues are
|
||||||
|
* computed.
|
||||||
* \returns Reference to \c *this
|
* \returns Reference to \c *this
|
||||||
*
|
*
|
||||||
* This function computes the eigenvalues and eigenvectors of \p matrix.
|
* This function computes the eigenvalues of the real matrix \p matrix.
|
||||||
* The eigenvalues() and eigenvectors() functions can be used to retrieve
|
* The eigenvalues() function can be used to retrieve them. If
|
||||||
* the computed eigendecomposition.
|
* \p computeEigenvectors is true, then the eigenvectors are also computed
|
||||||
|
* and can be retrieved by calling eigenvectors().
|
||||||
*
|
*
|
||||||
* The matrix is first reduced to real Schur form using the RealSchur
|
* The matrix is first reduced to real Schur form using the RealSchur
|
||||||
* class. The Schur decomposition is then used to compute the eigenvalues
|
* class. The Schur decomposition is then used to compute the eigenvalues
|
||||||
* and eigenvectors.
|
* and eigenvectors.
|
||||||
*
|
*
|
||||||
* The cost of the computation is dominated by the cost of the Schur
|
* The cost of the computation is dominated by the cost of the
|
||||||
* decomposition, which is very approximately \f$ 25n^3 \f$ where
|
* Schur decomposition, which is very approximately \f$ 25n^3 \f$
|
||||||
* \f$ n \f$ is the size of the matrix.
|
* (where \f$ n \f$ is the size of the matrix) if \p computeEigenvectors
|
||||||
|
* is true, and \f$ 10n^3 \f$ if \p computeEigenvectors is false.
|
||||||
*
|
*
|
||||||
* This method reuses of the allocated data in the EigenSolver object.
|
* This method reuses of the allocated data in the EigenSolver object.
|
||||||
*
|
*
|
||||||
* Example: \include EigenSolver_compute.cpp
|
* Example: \include EigenSolver_compute.cpp
|
||||||
* Output: \verbinclude EigenSolver_compute.out
|
* Output: \verbinclude EigenSolver_compute.out
|
||||||
*/
|
*/
|
||||||
EigenSolver& compute(const MatrixType& matrix);
|
EigenSolver& compute(const MatrixType& matrix, bool computeEigenvectors = true);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void computeEigenvectors();
|
void doComputeEigenvectors();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
MatrixType m_eivec;
|
MatrixType m_eivec;
|
||||||
EigenvalueType m_eivalues;
|
EigenvalueType m_eivalues;
|
||||||
bool m_isInitialized;
|
bool m_isInitialized;
|
||||||
|
bool m_eigenvectorsOk;
|
||||||
RealSchur<MatrixType> m_realSchur;
|
RealSchur<MatrixType> m_realSchur;
|
||||||
MatrixType m_matT;
|
MatrixType m_matT;
|
||||||
|
|
||||||
@ -286,7 +305,7 @@ template<typename MatrixType>
|
|||||||
MatrixType EigenSolver<MatrixType>::pseudoEigenvalueMatrix() const
|
MatrixType EigenSolver<MatrixType>::pseudoEigenvalueMatrix() const
|
||||||
{
|
{
|
||||||
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
||||||
Index n = m_eivec.cols();
|
Index n = m_eivalues.rows();
|
||||||
MatrixType matD = MatrixType::Zero(n,n);
|
MatrixType matD = MatrixType::Zero(n,n);
|
||||||
for (Index i=0; i<n; ++i)
|
for (Index i=0; i<n; ++i)
|
||||||
{
|
{
|
||||||
@ -306,6 +325,7 @@ template<typename MatrixType>
|
|||||||
typename EigenSolver<MatrixType>::EigenvectorsType EigenSolver<MatrixType>::eigenvectors() const
|
typename EigenSolver<MatrixType>::EigenvectorsType EigenSolver<MatrixType>::eigenvectors() const
|
||||||
{
|
{
|
||||||
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
ei_assert(m_isInitialized && "EigenSolver is not initialized.");
|
||||||
|
ei_assert(m_eigenvectorsOk && "The eigenvectors have not been computed together with the eigenvalues.");
|
||||||
Index n = m_eivec.cols();
|
Index n = m_eivec.cols();
|
||||||
EigenvectorsType matV(n,n);
|
EigenvectorsType matV(n,n);
|
||||||
for (Index j=0; j<n; ++j)
|
for (Index j=0; j<n; ++j)
|
||||||
@ -332,13 +352,14 @@ typename EigenSolver<MatrixType>::EigenvectorsType EigenSolver<MatrixType>::eige
|
|||||||
}
|
}
|
||||||
|
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
EigenSolver<MatrixType>& EigenSolver<MatrixType>::compute(const MatrixType& matrix)
|
EigenSolver<MatrixType>& EigenSolver<MatrixType>::compute(const MatrixType& matrix, bool computeEigenvectors)
|
||||||
{
|
{
|
||||||
assert(matrix.cols() == matrix.rows());
|
assert(matrix.cols() == matrix.rows());
|
||||||
|
|
||||||
// Reduce to real Schur form.
|
// Reduce to real Schur form.
|
||||||
m_realSchur.compute(matrix);
|
m_realSchur.compute(matrix, computeEigenvectors);
|
||||||
m_matT = m_realSchur.matrixT();
|
m_matT = m_realSchur.matrixT();
|
||||||
|
if (computeEigenvectors)
|
||||||
m_eivec = m_realSchur.matrixU();
|
m_eivec = m_realSchur.matrixU();
|
||||||
|
|
||||||
// Compute eigenvalues from matT
|
// Compute eigenvalues from matT
|
||||||
@ -362,9 +383,12 @@ EigenSolver<MatrixType>& EigenSolver<MatrixType>::compute(const MatrixType& matr
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Compute eigenvectors.
|
// Compute eigenvectors.
|
||||||
computeEigenvectors();
|
if (computeEigenvectors)
|
||||||
|
doComputeEigenvectors();
|
||||||
|
|
||||||
m_isInitialized = true;
|
m_isInitialized = true;
|
||||||
|
m_eigenvectorsOk = computeEigenvectors;
|
||||||
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -389,7 +413,7 @@ std::complex<Scalar> cdiv(Scalar xr, Scalar xi, Scalar yr, Scalar yi)
|
|||||||
|
|
||||||
|
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
void EigenSolver<MatrixType>::computeEigenvectors()
|
void EigenSolver<MatrixType>::doComputeEigenvectors()
|
||||||
{
|
{
|
||||||
const Index size = m_eivec.cols();
|
const Index size = m_eivec.cols();
|
||||||
const Scalar eps = NumTraits<Scalar>::epsilon();
|
const Scalar eps = NumTraits<Scalar>::epsilon();
|
||||||
|
@ -37,7 +37,7 @@ struct ei_eigenvalues_selector
|
|||||||
{
|
{
|
||||||
typedef typename Derived::PlainObject PlainObject;
|
typedef typename Derived::PlainObject PlainObject;
|
||||||
PlainObject m_eval(m);
|
PlainObject m_eval(m);
|
||||||
return ComplexEigenSolver<PlainObject>(m_eval).eigenvalues();
|
return ComplexEigenSolver<PlainObject>(m_eval, false).eigenvalues();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -49,7 +49,7 @@ struct ei_eigenvalues_selector<Derived, false>
|
|||||||
{
|
{
|
||||||
typedef typename Derived::PlainObject PlainObject;
|
typedef typename Derived::PlainObject PlainObject;
|
||||||
PlainObject m_eval(m);
|
PlainObject m_eval(m);
|
||||||
return EigenSolver<PlainObject>(m_eval).eigenvalues();
|
return EigenSolver<PlainObject>(m_eval, false).eigenvalues();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -101,7 +101,7 @@ SelfAdjointView<MatrixType, UpLo>::eigenvalues() const
|
|||||||
{
|
{
|
||||||
typedef typename SelfAdjointView<MatrixType, UpLo>::PlainObject PlainObject;
|
typedef typename SelfAdjointView<MatrixType, UpLo>::PlainObject PlainObject;
|
||||||
PlainObject thisAsMatrix(*this);
|
PlainObject thisAsMatrix(*this);
|
||||||
return SelfAdjointEigenSolver<PlainObject>(thisAsMatrix).eigenvalues();
|
return SelfAdjointEigenSolver<PlainObject>(thisAsMatrix, false).eigenvalues();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,13 +50,13 @@
|
|||||||
* the eigendecomposition of a matrix.
|
* the eigendecomposition of a matrix.
|
||||||
*
|
*
|
||||||
* Call the function compute() to compute the real Schur decomposition of a
|
* Call the function compute() to compute the real Schur decomposition of a
|
||||||
* given matrix. Alternatively, you can use the RealSchur(const MatrixType&)
|
* given matrix. Alternatively, you can use the RealSchur(const MatrixType&, bool)
|
||||||
* constructor which computes the real Schur decomposition at construction
|
* constructor which computes the real Schur decomposition at construction
|
||||||
* time. Once the decomposition is computed, you can use the matrixU() and
|
* time. Once the decomposition is computed, you can use the matrixU() and
|
||||||
* matrixT() functions to retrieve the matrices U and T in the decomposition.
|
* matrixT() functions to retrieve the matrices U and T in the decomposition.
|
||||||
*
|
*
|
||||||
* The documentation of RealSchur(const MatrixType&) contains an example of
|
* The documentation of RealSchur(const MatrixType&, bool) contains an example
|
||||||
* the typical use of this class.
|
* of the typical use of this class.
|
||||||
*
|
*
|
||||||
* \note The implementation is adapted from
|
* \note The implementation is adapted from
|
||||||
* <a href="http://math.nist.gov/javanumerics/jama/">JAMA</a> (public domain).
|
* <a href="http://math.nist.gov/javanumerics/jama/">JAMA</a> (public domain).
|
||||||
@ -98,41 +98,46 @@ template<typename _MatrixType> class RealSchur
|
|||||||
m_matU(size, size),
|
m_matU(size, size),
|
||||||
m_workspaceVector(size),
|
m_workspaceVector(size),
|
||||||
m_hess(size),
|
m_hess(size),
|
||||||
m_isInitialized(false)
|
m_isInitialized(false),
|
||||||
|
m_matUisUptodate(false)
|
||||||
{ }
|
{ }
|
||||||
|
|
||||||
/** \brief Constructor; computes real Schur decomposition of given matrix.
|
/** \brief Constructor; computes real Schur decomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose Schur decomposition is to be computed.
|
* \param[in] matrix Square matrix whose Schur decomposition is to be computed.
|
||||||
|
* \param[in] computeU If true, both T and U are computed; if false, only T is computed.
|
||||||
*
|
*
|
||||||
* This constructor calls compute() to compute the Schur decomposition.
|
* This constructor calls compute() to compute the Schur decomposition.
|
||||||
*
|
*
|
||||||
* Example: \include RealSchur_RealSchur_MatrixType.cpp
|
* Example: \include RealSchur_RealSchur_MatrixType.cpp
|
||||||
* Output: \verbinclude RealSchur_RealSchur_MatrixType.out
|
* Output: \verbinclude RealSchur_RealSchur_MatrixType.out
|
||||||
*/
|
*/
|
||||||
RealSchur(const MatrixType& matrix)
|
RealSchur(const MatrixType& matrix, bool computeU = true)
|
||||||
: m_matT(matrix.rows(),matrix.cols()),
|
: m_matT(matrix.rows(),matrix.cols()),
|
||||||
m_matU(matrix.rows(),matrix.cols()),
|
m_matU(matrix.rows(),matrix.cols()),
|
||||||
m_workspaceVector(matrix.rows()),
|
m_workspaceVector(matrix.rows()),
|
||||||
m_hess(matrix.rows()),
|
m_hess(matrix.rows()),
|
||||||
m_isInitialized(false)
|
m_isInitialized(false),
|
||||||
|
m_matUisUptodate(false)
|
||||||
{
|
{
|
||||||
compute(matrix);
|
compute(matrix, computeU);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Returns the orthogonal matrix in the Schur decomposition.
|
/** \brief Returns the orthogonal matrix in the Schur decomposition.
|
||||||
*
|
*
|
||||||
* \returns A const reference to the matrix U.
|
* \returns A const reference to the matrix U.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor RealSchur(const MatrixType&) or the member
|
* \pre Either the constructor RealSchur(const MatrixType&, bool) or the
|
||||||
* function compute(const MatrixType&) has been called before to compute
|
* member function compute(const MatrixType&, bool) has been called before
|
||||||
* the Schur decomposition of a matrix.
|
* to compute the Schur decomposition of a matrix, and \p computeU was set
|
||||||
|
* to true (the default value).
|
||||||
*
|
*
|
||||||
* \sa RealSchur(const MatrixType&) for an example
|
* \sa RealSchur(const MatrixType&, bool) for an example
|
||||||
*/
|
*/
|
||||||
const MatrixType& matrixU() const
|
const MatrixType& matrixU() const
|
||||||
{
|
{
|
||||||
ei_assert(m_isInitialized && "RealSchur is not initialized.");
|
ei_assert(m_isInitialized && "RealSchur is not initialized.");
|
||||||
|
ei_assert(m_matUisUptodate && "The matrix U has not been computed during the RealSchur decomposition.");
|
||||||
return m_matU;
|
return m_matU;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,11 +145,11 @@ template<typename _MatrixType> class RealSchur
|
|||||||
*
|
*
|
||||||
* \returns A const reference to the matrix T.
|
* \returns A const reference to the matrix T.
|
||||||
*
|
*
|
||||||
* \pre Either the constructor RealSchur(const MatrixType&) or the member
|
* \pre Either the constructor RealSchur(const MatrixType&, bool) or the
|
||||||
* function compute(const MatrixType&) has been called before to compute
|
* member function compute(const MatrixType&, bool) has been called before
|
||||||
* the Schur decomposition of a matrix.
|
* to compute the Schur decomposition of a matrix.
|
||||||
*
|
*
|
||||||
* \sa RealSchur(const MatrixType&) for an example
|
* \sa RealSchur(const MatrixType&, bool) for an example
|
||||||
*/
|
*/
|
||||||
const MatrixType& matrixT() const
|
const MatrixType& matrixT() const
|
||||||
{
|
{
|
||||||
@ -155,18 +160,20 @@ template<typename _MatrixType> class RealSchur
|
|||||||
/** \brief Computes Schur decomposition of given matrix.
|
/** \brief Computes Schur decomposition of given matrix.
|
||||||
*
|
*
|
||||||
* \param[in] matrix Square matrix whose Schur decomposition is to be computed.
|
* \param[in] matrix Square matrix whose Schur decomposition is to be computed.
|
||||||
|
* \param[in] computeU If true, both T and U are computed; if false, only T is computed.
|
||||||
*
|
*
|
||||||
* The Schur decomposition is computed by first reducing the matrix to
|
* The Schur decomposition is computed by first reducing the matrix to
|
||||||
* Hessenberg form using the class HessenbergDecomposition. The Hessenberg
|
* Hessenberg form using the class HessenbergDecomposition. The Hessenberg
|
||||||
* matrix is then reduced to triangular form by performing Francis QR
|
* matrix is then reduced to triangular form by performing Francis QR
|
||||||
* iterations with implicit double shift. The cost of computing the Schur
|
* iterations with implicit double shift. The cost of computing the Schur
|
||||||
* decomposition depends on the number of iterations; as a rough guide, it
|
* decomposition depends on the number of iterations; as a rough guide, it
|
||||||
* may be taken to be \f$25n^3\f$ flops.
|
* may be taken to be \f$25n^3\f$ flops if \a computeU is true and
|
||||||
|
* \f$10n^3\f$ flops if \a computeU is false.
|
||||||
*
|
*
|
||||||
* Example: \include RealSchur_compute.cpp
|
* Example: \include RealSchur_compute.cpp
|
||||||
* Output: \verbinclude RealSchur_compute.out
|
* Output: \verbinclude RealSchur_compute.out
|
||||||
*/
|
*/
|
||||||
void compute(const MatrixType& matrix);
|
void compute(const MatrixType& matrix, bool computeU = true);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
@ -175,38 +182,39 @@ template<typename _MatrixType> class RealSchur
|
|||||||
ColumnVectorType m_workspaceVector;
|
ColumnVectorType m_workspaceVector;
|
||||||
HessenbergDecomposition<MatrixType> m_hess;
|
HessenbergDecomposition<MatrixType> m_hess;
|
||||||
bool m_isInitialized;
|
bool m_isInitialized;
|
||||||
|
bool m_matUisUptodate;
|
||||||
|
|
||||||
typedef Matrix<Scalar,3,1> Vector3s;
|
typedef Matrix<Scalar,3,1> Vector3s;
|
||||||
|
|
||||||
Scalar computeNormOfT();
|
Scalar computeNormOfT();
|
||||||
Index findSmallSubdiagEntry(Index iu, Scalar norm);
|
Index findSmallSubdiagEntry(Index iu, Scalar norm);
|
||||||
void splitOffTwoRows(Index iu, Scalar exshift);
|
void splitOffTwoRows(Index iu, bool computeU, Scalar exshift);
|
||||||
void computeShift(Index iu, Index iter, Scalar& exshift, Vector3s& shiftInfo);
|
void computeShift(Index iu, Index iter, Scalar& exshift, Vector3s& shiftInfo);
|
||||||
void initFrancisQRStep(Index il, Index iu, const Vector3s& shiftInfo, Index& im, Vector3s& firstHouseholderVector);
|
void initFrancisQRStep(Index il, Index iu, const Vector3s& shiftInfo, Index& im, Vector3s& firstHouseholderVector);
|
||||||
void performFrancisQRStep(Index il, Index im, Index iu, const Vector3s& firstHouseholderVector, Scalar* workspace);
|
void performFrancisQRStep(Index il, Index im, Index iu, bool computeU, const Vector3s& firstHouseholderVector, Scalar* workspace);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
void RealSchur<MatrixType>::compute(const MatrixType& matrix)
|
void RealSchur<MatrixType>::compute(const MatrixType& matrix, bool computeU)
|
||||||
{
|
{
|
||||||
assert(matrix.cols() == matrix.rows());
|
assert(matrix.cols() == matrix.rows());
|
||||||
|
|
||||||
// Step 1. Reduce to Hessenberg form
|
// Step 1. Reduce to Hessenberg form
|
||||||
// TODO skip Q if skipU = true
|
|
||||||
m_hess.compute(matrix);
|
m_hess.compute(matrix);
|
||||||
m_matT = m_hess.matrixH();
|
m_matT = m_hess.matrixH();
|
||||||
|
if (computeU)
|
||||||
m_matU = m_hess.matrixQ();
|
m_matU = m_hess.matrixQ();
|
||||||
|
|
||||||
// Step 2. Reduce to real Schur form
|
// Step 2. Reduce to real Schur form
|
||||||
m_workspaceVector.resize(m_matU.cols());
|
m_workspaceVector.resize(m_matT.cols());
|
||||||
Scalar* workspace = &m_workspaceVector.coeffRef(0);
|
Scalar* workspace = &m_workspaceVector.coeffRef(0);
|
||||||
|
|
||||||
// The matrix m_matT is divided in three parts.
|
// The matrix m_matT is divided in three parts.
|
||||||
// Rows 0,...,il-1 are decoupled from the rest because m_matT(il,il-1) is zero.
|
// Rows 0,...,il-1 are decoupled from the rest because m_matT(il,il-1) is zero.
|
||||||
// Rows il,...,iu is the part we are working on (the active window).
|
// Rows il,...,iu is the part we are working on (the active window).
|
||||||
// Rows iu+1,...,end are already brought in triangular form.
|
// Rows iu+1,...,end are already brought in triangular form.
|
||||||
Index iu = m_matU.cols() - 1;
|
Index iu = m_matT.cols() - 1;
|
||||||
Index iter = 0; // iteration count
|
Index iter = 0; // iteration count
|
||||||
Scalar exshift = 0.0; // sum of exceptional shifts
|
Scalar exshift = 0.0; // sum of exceptional shifts
|
||||||
Scalar norm = computeNormOfT();
|
Scalar norm = computeNormOfT();
|
||||||
@ -226,7 +234,7 @@ void RealSchur<MatrixType>::compute(const MatrixType& matrix)
|
|||||||
}
|
}
|
||||||
else if (il == iu-1) // Two roots found
|
else if (il == iu-1) // Two roots found
|
||||||
{
|
{
|
||||||
splitOffTwoRows(iu, exshift);
|
splitOffTwoRows(iu, computeU, exshift);
|
||||||
iu -= 2;
|
iu -= 2;
|
||||||
iter = 0;
|
iter = 0;
|
||||||
}
|
}
|
||||||
@ -237,18 +245,19 @@ void RealSchur<MatrixType>::compute(const MatrixType& matrix)
|
|||||||
iter = iter + 1; // (Could check iteration count here.)
|
iter = iter + 1; // (Could check iteration count here.)
|
||||||
Index im;
|
Index im;
|
||||||
initFrancisQRStep(il, iu, shiftInfo, im, firstHouseholderVector);
|
initFrancisQRStep(il, iu, shiftInfo, im, firstHouseholderVector);
|
||||||
performFrancisQRStep(il, im, iu, firstHouseholderVector, workspace);
|
performFrancisQRStep(il, im, iu, computeU, firstHouseholderVector, workspace);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_isInitialized = true;
|
m_isInitialized = true;
|
||||||
|
m_matUisUptodate = computeU;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \internal Computes and returns vector L1 norm of T */
|
/** \internal Computes and returns vector L1 norm of T */
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
inline typename MatrixType::Scalar RealSchur<MatrixType>::computeNormOfT()
|
inline typename MatrixType::Scalar RealSchur<MatrixType>::computeNormOfT()
|
||||||
{
|
{
|
||||||
const Index size = m_matU.cols();
|
const Index size = m_matT.cols();
|
||||||
// FIXME to be efficient the following would requires a triangular reduxion code
|
// FIXME to be efficient the following would requires a triangular reduxion code
|
||||||
// Scalar norm = m_matT.upper().cwiseAbs().sum()
|
// Scalar norm = m_matT.upper().cwiseAbs().sum()
|
||||||
// + m_matT.bottomLeftCorner(size-1,size-1).diagonal().cwiseAbs().sum();
|
// + m_matT.bottomLeftCorner(size-1,size-1).diagonal().cwiseAbs().sum();
|
||||||
@ -277,9 +286,9 @@ inline typename MatrixType::Index RealSchur<MatrixType>::findSmallSubdiagEntry(I
|
|||||||
|
|
||||||
/** \internal Update T given that rows iu-1 and iu decouple from the rest. */
|
/** \internal Update T given that rows iu-1 and iu decouple from the rest. */
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
inline void RealSchur<MatrixType>::splitOffTwoRows(Index iu, Scalar exshift)
|
inline void RealSchur<MatrixType>::splitOffTwoRows(Index iu, bool computeU, Scalar exshift)
|
||||||
{
|
{
|
||||||
const Index size = m_matU.cols();
|
const Index size = m_matT.cols();
|
||||||
|
|
||||||
// The eigenvalues of the 2x2 matrix [a b; c d] are
|
// The eigenvalues of the 2x2 matrix [a b; c d] are
|
||||||
// trace +/- sqrt(discr/4) where discr = tr^2 - 4*det, tr = a + d, det = ad - bc
|
// trace +/- sqrt(discr/4) where discr = tr^2 - 4*det, tr = a + d, det = ad - bc
|
||||||
@ -300,6 +309,7 @@ inline void RealSchur<MatrixType>::splitOffTwoRows(Index iu, Scalar exshift)
|
|||||||
m_matT.rightCols(size-iu+1).applyOnTheLeft(iu-1, iu, rot.adjoint());
|
m_matT.rightCols(size-iu+1).applyOnTheLeft(iu-1, iu, rot.adjoint());
|
||||||
m_matT.topRows(iu+1).applyOnTheRight(iu-1, iu, rot);
|
m_matT.topRows(iu+1).applyOnTheRight(iu-1, iu, rot);
|
||||||
m_matT.coeffRef(iu, iu-1) = Scalar(0);
|
m_matT.coeffRef(iu, iu-1) = Scalar(0);
|
||||||
|
if (computeU)
|
||||||
m_matU.applyOnTheRight(iu-1, iu, rot);
|
m_matU.applyOnTheRight(iu-1, iu, rot);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -375,12 +385,12 @@ inline void RealSchur<MatrixType>::initFrancisQRStep(Index il, Index iu, const V
|
|||||||
|
|
||||||
/** \internal Perform a Francis QR step involving rows il:iu and columns im:iu. */
|
/** \internal Perform a Francis QR step involving rows il:iu and columns im:iu. */
|
||||||
template<typename MatrixType>
|
template<typename MatrixType>
|
||||||
inline void RealSchur<MatrixType>::performFrancisQRStep(Index il, Index im, Index iu, const Vector3s& firstHouseholderVector, Scalar* workspace)
|
inline void RealSchur<MatrixType>::performFrancisQRStep(Index il, Index im, Index iu, bool computeU, const Vector3s& firstHouseholderVector, Scalar* workspace)
|
||||||
{
|
{
|
||||||
assert(im >= il);
|
assert(im >= il);
|
||||||
assert(im <= iu-2);
|
assert(im <= iu-2);
|
||||||
|
|
||||||
const Index size = m_matU.cols();
|
const Index size = m_matT.cols();
|
||||||
|
|
||||||
for (Index k = im; k <= iu-2; ++k)
|
for (Index k = im; k <= iu-2; ++k)
|
||||||
{
|
{
|
||||||
@ -406,6 +416,7 @@ inline void RealSchur<MatrixType>::performFrancisQRStep(Index il, Index im, Inde
|
|||||||
// These Householder transformations form the O(n^3) part of the algorithm
|
// These Householder transformations form the O(n^3) part of the algorithm
|
||||||
m_matT.block(k, k, 3, size-k).applyHouseholderOnTheLeft(ess, tau, workspace);
|
m_matT.block(k, k, 3, size-k).applyHouseholderOnTheLeft(ess, tau, workspace);
|
||||||
m_matT.block(0, k, std::min(iu,k+3) + 1, 3).applyHouseholderOnTheRight(ess, tau, workspace);
|
m_matT.block(0, k, std::min(iu,k+3) + 1, 3).applyHouseholderOnTheRight(ess, tau, workspace);
|
||||||
|
if (computeU)
|
||||||
m_matU.block(0, k, size, 3).applyHouseholderOnTheRight(ess, tau, workspace);
|
m_matU.block(0, k, size, 3).applyHouseholderOnTheRight(ess, tau, workspace);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -420,6 +431,7 @@ inline void RealSchur<MatrixType>::performFrancisQRStep(Index il, Index im, Inde
|
|||||||
m_matT.coeffRef(iu-1, iu-2) = beta;
|
m_matT.coeffRef(iu-1, iu-2) = beta;
|
||||||
m_matT.block(iu-1, iu-1, 2, size-iu+1).applyHouseholderOnTheLeft(ess, tau, workspace);
|
m_matT.block(iu-1, iu-1, 2, size-iu+1).applyHouseholderOnTheLeft(ess, tau, workspace);
|
||||||
m_matT.block(0, iu-1, iu+1, 2).applyHouseholderOnTheRight(ess, tau, workspace);
|
m_matT.block(0, iu-1, iu+1, 2).applyHouseholderOnTheRight(ess, tau, workspace);
|
||||||
|
if (computeU)
|
||||||
m_matU.block(0, iu-1, size, 2).applyHouseholderOnTheRight(ess, tau, workspace);
|
m_matU.block(0, iu-1, size, 2).applyHouseholderOnTheRight(ess, tau, workspace);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
MatrixXcf ones = MatrixXcf::Ones(3,3);
|
MatrixXcf ones = MatrixXcf::Ones(3,3);
|
||||||
ComplexEigenSolver<MatrixXcf> ces(ones);
|
ComplexEigenSolver<MatrixXcf> ces(ones, /* computeEigenvectors = */ false);
|
||||||
cout << "The eigenvalues of the 3x3 matrix of ones are:"
|
cout << "The eigenvalues of the 3x3 matrix of ones are:"
|
||||||
<< endl << ces.eigenvalues() << endl;
|
<< endl << ces.eigenvalues() << endl;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
EigenSolver<MatrixXf> es;
|
EigenSolver<MatrixXf> es;
|
||||||
MatrixXf A = MatrixXf::Random(4,4);
|
MatrixXf A = MatrixXf::Random(4,4);
|
||||||
es.compute(A);
|
es.compute(A, /* computeEigenvectors = */ false);
|
||||||
cout << "The eigenvalues of A are: " << es.eigenvalues().transpose() << endl;
|
cout << "The eigenvalues of A are: " << es.eigenvalues().transpose() << endl;
|
||||||
es.compute(A + MatrixXf::Identity(4,4)); // re-use es to compute eigenvalues of A+I
|
es.compute(A + MatrixXf::Identity(4,4), false); // re-use es to compute eigenvalues of A+I
|
||||||
cout << "The eigenvalues of A+I are: " << es.eigenvalues().transpose() << endl;
|
cout << "The eigenvalues of A+I are: " << es.eigenvalues().transpose() << endl;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
MatrixXd ones = MatrixXd::Ones(3,3);
|
MatrixXd ones = MatrixXd::Ones(3,3);
|
||||||
EigenSolver<MatrixXd> es(ones);
|
EigenSolver<MatrixXd> es(ones, false);
|
||||||
cout << "The eigenvalues of the 3x3 matrix of ones are:"
|
cout << "The eigenvalues of the 3x3 matrix of ones are:"
|
||||||
<< endl << es.eigenvalues() << endl;
|
<< endl << es.eigenvalues() << endl;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
MatrixXf A = MatrixXf::Random(4,4);
|
MatrixXf A = MatrixXf::Random(4,4);
|
||||||
RealSchur<MatrixXf> schur(4);
|
RealSchur<MatrixXf> schur(4);
|
||||||
schur.compute(A);
|
schur.compute(A, /* computeU = */ false);
|
||||||
cout << "The matrix T in the decomposition of A is:" << endl << schur.matrixT() << endl;
|
cout << "The matrix T in the decomposition of A is:" << endl << schur.matrixT() << endl;
|
||||||
schur.compute(A.inverse());
|
schur.compute(A.inverse(), /* computeU = */ false);
|
||||||
cout << "The matrix T in the decomposition of A^(-1) is:" << endl << schur.matrixT() << endl;
|
cout << "The matrix T in the decomposition of A^(-1) is:" << endl << schur.matrixT() << endl;
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
// for linear algebra. Eigen itself is part of the KDE project.
|
// for linear algebra. Eigen itself is part of the KDE project.
|
||||||
//
|
//
|
||||||
// Copyright (C) 2008-2009 Gael Guennebaud <g.gael@free.fr>
|
// Copyright (C) 2008-2009 Gael Guennebaud <g.gael@free.fr>
|
||||||
|
// Copyright (C) 2010 Jitse Niesen <jitse@maths.leeds.ac.uk>
|
||||||
//
|
//
|
||||||
// Eigen is free software; you can redistribute it and/or
|
// Eigen is free software; you can redistribute it and/or
|
||||||
// modify it under the terms of the GNU Lesser General Public
|
// modify it under the terms of the GNU Lesser General Public
|
||||||
@ -67,6 +68,9 @@ template<typename MatrixType> void eigensolver(const MatrixType& m)
|
|||||||
// another algorithm so results may differ slightly
|
// another algorithm so results may differ slightly
|
||||||
verify_is_approx_upto_permutation(a.eigenvalues(), ei1.eigenvalues());
|
verify_is_approx_upto_permutation(a.eigenvalues(), ei1.eigenvalues());
|
||||||
|
|
||||||
|
ComplexEigenSolver<MatrixType> eiNoEivecs(a, false);
|
||||||
|
VERIFY_IS_APPROX(ei1.eigenvalues(), eiNoEivecs.eigenvalues());
|
||||||
|
|
||||||
// Regression test for issue #66
|
// Regression test for issue #66
|
||||||
MatrixType z = MatrixType::Zero(rows,cols);
|
MatrixType z = MatrixType::Zero(rows,cols);
|
||||||
ComplexEigenSolver<MatrixType> eiz(z);
|
ComplexEigenSolver<MatrixType> eiz(z);
|
||||||
@ -76,11 +80,15 @@ template<typename MatrixType> void eigensolver(const MatrixType& m)
|
|||||||
VERIFY_IS_APPROX(id.operatorNorm(), RealScalar(1));
|
VERIFY_IS_APPROX(id.operatorNorm(), RealScalar(1));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename MatrixType> void eigensolver_verify_assert()
|
template<typename MatrixType> void eigensolver_verify_assert(const MatrixType& m)
|
||||||
{
|
{
|
||||||
ComplexEigenSolver<MatrixType> eig;
|
ComplexEigenSolver<MatrixType> eig;
|
||||||
VERIFY_RAISES_ASSERT(eig.eigenvectors())
|
VERIFY_RAISES_ASSERT(eig.eigenvectors());
|
||||||
VERIFY_RAISES_ASSERT(eig.eigenvalues())
|
VERIFY_RAISES_ASSERT(eig.eigenvalues());
|
||||||
|
|
||||||
|
MatrixType a = MatrixType::Random(m.rows(),m.cols());
|
||||||
|
eig.compute(a, false);
|
||||||
|
VERIFY_RAISES_ASSERT(eig.eigenvectors());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_eigensolver_complex()
|
void test_eigensolver_complex()
|
||||||
@ -92,10 +100,10 @@ void test_eigensolver_complex()
|
|||||||
CALL_SUBTEST_4( eigensolver(Matrix3f()) );
|
CALL_SUBTEST_4( eigensolver(Matrix3f()) );
|
||||||
}
|
}
|
||||||
|
|
||||||
CALL_SUBTEST_1( eigensolver_verify_assert<Matrix4cf>() );
|
CALL_SUBTEST_1( eigensolver_verify_assert(Matrix4cf()) );
|
||||||
CALL_SUBTEST_2( eigensolver_verify_assert<MatrixXcd>() );
|
CALL_SUBTEST_2( eigensolver_verify_assert(MatrixXcd(14,14)) );
|
||||||
CALL_SUBTEST_3(( eigensolver_verify_assert<Matrix<std::complex<float>, 1, 1> >() ));
|
CALL_SUBTEST_3( eigensolver_verify_assert(Matrix<std::complex<float>, 1, 1>()) );
|
||||||
CALL_SUBTEST_4( eigensolver_verify_assert<Matrix3f>() );
|
CALL_SUBTEST_4( eigensolver_verify_assert(Matrix3f()) );
|
||||||
|
|
||||||
// Test problem size constructors
|
// Test problem size constructors
|
||||||
CALL_SUBTEST_5(ComplexEigenSolver<MatrixXf>(10));
|
CALL_SUBTEST_5(ComplexEigenSolver<MatrixXf>(10));
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
// for linear algebra.
|
// for linear algebra.
|
||||||
//
|
//
|
||||||
// Copyright (C) 2008 Gael Guennebaud <g.gael@free.fr>
|
// Copyright (C) 2008 Gael Guennebaud <g.gael@free.fr>
|
||||||
|
// Copyright (C) 2010 Jitse Niesen <jitse@maths.leeds.ac.uk>
|
||||||
//
|
//
|
||||||
// Eigen is free software; you can redistribute it and/or
|
// Eigen is free software; you can redistribute it and/or
|
||||||
// modify it under the terms of the GNU Lesser General Public
|
// modify it under the terms of the GNU Lesser General Public
|
||||||
@ -60,19 +61,26 @@ template<typename MatrixType> void eigensolver(const MatrixType& m)
|
|||||||
ei1.eigenvectors() * ei1.eigenvalues().asDiagonal());
|
ei1.eigenvectors() * ei1.eigenvalues().asDiagonal());
|
||||||
VERIFY_IS_APPROX(a.eigenvalues(), ei1.eigenvalues());
|
VERIFY_IS_APPROX(a.eigenvalues(), ei1.eigenvalues());
|
||||||
|
|
||||||
|
EigenSolver<MatrixType> eiNoEivecs(a, false);
|
||||||
|
VERIFY_IS_APPROX(ei1.eigenvalues(), eiNoEivecs.eigenvalues());
|
||||||
|
VERIFY_IS_APPROX(ei1.pseudoEigenvalueMatrix(), eiNoEivecs.pseudoEigenvalueMatrix());
|
||||||
|
|
||||||
MatrixType id = MatrixType::Identity(rows, cols);
|
MatrixType id = MatrixType::Identity(rows, cols);
|
||||||
VERIFY_IS_APPROX(id.operatorNorm(), RealScalar(1));
|
VERIFY_IS_APPROX(id.operatorNorm(), RealScalar(1));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename MatrixType> void eigensolver_verify_assert()
|
template<typename MatrixType> void eigensolver_verify_assert(const MatrixType& m)
|
||||||
{
|
{
|
||||||
MatrixType tmp;
|
|
||||||
|
|
||||||
EigenSolver<MatrixType> eig;
|
EigenSolver<MatrixType> eig;
|
||||||
VERIFY_RAISES_ASSERT(eig.eigenvectors())
|
VERIFY_RAISES_ASSERT(eig.eigenvectors());
|
||||||
VERIFY_RAISES_ASSERT(eig.pseudoEigenvectors())
|
VERIFY_RAISES_ASSERT(eig.pseudoEigenvectors());
|
||||||
VERIFY_RAISES_ASSERT(eig.pseudoEigenvalueMatrix())
|
VERIFY_RAISES_ASSERT(eig.pseudoEigenvalueMatrix());
|
||||||
VERIFY_RAISES_ASSERT(eig.eigenvalues())
|
VERIFY_RAISES_ASSERT(eig.eigenvalues());
|
||||||
|
|
||||||
|
MatrixType a = MatrixType::Random(m.rows(),m.cols());
|
||||||
|
eig.compute(a, false);
|
||||||
|
VERIFY_RAISES_ASSERT(eig.eigenvectors());
|
||||||
|
VERIFY_RAISES_ASSERT(eig.pseudoEigenvectors());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_eigensolver_generic()
|
void test_eigensolver_generic()
|
||||||
@ -88,11 +96,11 @@ void test_eigensolver_generic()
|
|||||||
CALL_SUBTEST_4( eigensolver(Matrix2d()) );
|
CALL_SUBTEST_4( eigensolver(Matrix2d()) );
|
||||||
}
|
}
|
||||||
|
|
||||||
CALL_SUBTEST_1( eigensolver_verify_assert<Matrix4f>() );
|
CALL_SUBTEST_1( eigensolver_verify_assert(Matrix4f()) );
|
||||||
CALL_SUBTEST_2( eigensolver_verify_assert<MatrixXd>() );
|
CALL_SUBTEST_2( eigensolver_verify_assert(MatrixXd(17,17)) );
|
||||||
CALL_SUBTEST_4( eigensolver_verify_assert<Matrix2d>() );
|
CALL_SUBTEST_3( eigensolver_verify_assert(Matrix<double,1,1>()) );
|
||||||
CALL_SUBTEST_5( eigensolver_verify_assert<MatrixXf>() );
|
CALL_SUBTEST_4( eigensolver_verify_assert(Matrix2d()) );
|
||||||
|
|
||||||
// Test problem size constructors
|
// Test problem size constructors
|
||||||
CALL_SUBTEST_6(EigenSolver<MatrixXf>(10));
|
CALL_SUBTEST_5(EigenSolver<MatrixXf>(10));
|
||||||
}
|
}
|
||||||
|
@ -73,6 +73,11 @@ template<typename MatrixType> void schur(int size = MatrixType::ColsAtCompileTim
|
|||||||
RealSchur<MatrixType> rs2(A);
|
RealSchur<MatrixType> rs2(A);
|
||||||
VERIFY_IS_EQUAL(rs1.matrixT(), rs2.matrixT());
|
VERIFY_IS_EQUAL(rs1.matrixT(), rs2.matrixT());
|
||||||
VERIFY_IS_EQUAL(rs1.matrixU(), rs2.matrixU());
|
VERIFY_IS_EQUAL(rs1.matrixU(), rs2.matrixU());
|
||||||
|
|
||||||
|
// Test computation of only T, not U
|
||||||
|
RealSchur<MatrixType> rsOnlyT(A, false);
|
||||||
|
VERIFY_IS_EQUAL(rs1.matrixT(), rsOnlyT.matrixT());
|
||||||
|
VERIFY_RAISES_ASSERT(rsOnlyT.matrixU());
|
||||||
}
|
}
|
||||||
|
|
||||||
void test_schur_real()
|
void test_schur_real()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user