From 24db2833154c122769c95e17ab6be1e8b7f520cd Mon Sep 17 00:00:00 2001 From: nyoungbq Date: Fri, 20 Dec 2024 16:49:02 -0500 Subject: [PATCH] correct production of eigenvalues/vectors --- .../Algorithms/ComputeTriangleGeomShapes.cpp | 53 +++++++++++++------ 1 file changed, 36 insertions(+), 17 deletions(-) diff --git a/src/Plugins/OrientationAnalysis/src/OrientationAnalysis/Filters/Algorithms/ComputeTriangleGeomShapes.cpp b/src/Plugins/OrientationAnalysis/src/OrientationAnalysis/Filters/Algorithms/ComputeTriangleGeomShapes.cpp index 5c8d985075..84bba9957f 100644 --- a/src/Plugins/OrientationAnalysis/src/OrientationAnalysis/Filters/Algorithms/ComputeTriangleGeomShapes.cpp +++ b/src/Plugins/OrientationAnalysis/src/OrientationAnalysis/Filters/Algorithms/ComputeTriangleGeomShapes.cpp @@ -175,39 +175,58 @@ void ComputeTriangleGeomShapes::findMoments() * - QR Factorization is different than QR Decomposition. * - QR Decomposition expresses the product of an Orthogonal Matrix (Q) and an right/upper triangular matrix (R) as a singular matrix (A). */ + + // Zero out small numbers in Cinertia: https://stackoverflow.com/a/54505281 + Cinertia = (0.000000001 < Cinertia.array().abs()).select(Cinertia, 0.0); + // TODO: // - Remove Copying between processing (inline integration) // - Extract real part of complexes stored in eigenvalues/vectors Eigen::HessenbergDecomposition hessDecomp(Cinertia); Matrix3x3 hessenMatrixUpper = hessDecomp.matrixH(); - Eigen::HouseholderQR hQR(hessenMatrixUpper); + // It is important to maintain the FullPivHouseholderQR implementation here, it is rank preserving and most numerically stable + Eigen::FullPivHouseholderQR hQR(hessenMatrixUpper); Matrix3x3 hqrMatrix = hQR.matrixQR(); // Extract eigenvalues and eigenvectors Eigen::EigenSolver eigenSolver(hqrMatrix); - // The primary axis is the largest eigenvector + // The primary axis is the largest eigenvalue Eigen::EigenSolver::EigenvalueType eigenvalues = eigenSolver.eigenvalues(); // This is the angular velocity vector, each row represents an axial alignment (principle axis) Eigen::EigenSolver::EigenvectorsType eigenvectors = eigenSolver.eigenvectors(); - std::cout << "Eigenvalues:\n" << eigenvalues << std::endl; - std::cout << "\n Eigenvectors:\n" << eigenvectors << std::endl; - - constexpr char k_BaselineAxisLabel = 'x'; // x - char axisLabel = 'x'; - double primaryAxis = eigenvalues[0].real(); - for(usize i = 1; i < eigenvalues.size(); i++) - { - if(primaryAxis < eigenvalues[i].real()) - { - axisLabel = k_BaselineAxisLabel + static_cast(i); - primaryAxis = eigenvalues[i].real(); - } - } - std::cout << "\nPrimary Axis: " << axisLabel << " | Associated Eigenvalue: " << primaryAxis << std::endl; + /** + * Following section for debugging + */ + // std::cout << "Eigenvalues:\n" << eigenvalues << std::endl; + // std::cout << "\n Eigenvectors:\n" << eigenvectors << std::endl; + // + // constexpr char k_BaselineAxisLabel = 'x'; // x + // char axisLabel = 'x'; + // double primaryAxis = eigenvalues[0].real(); + // for(usize i = 1; i < eigenvalues.size(); i++) + // { + // if(primaryAxis < eigenvalues[i].real()) + // { + // axisLabel = k_BaselineAxisLabel + static_cast(i); + // primaryAxis = eigenvalues[i].real(); + // } + // } + // std::cout << "\nPrimary Axis: " << axisLabel << " | Associated Eigenvalue: " << primaryAxis << std::endl; + // + // // Condition Number calculations from here: https://stackoverflow.com/questions/33575478/how-can-you-find-the-condition-number-in-eigen + // // Calculate Condition Number from JacobiSVD + // Eigen::JacobiSVD svd(hqrMatrix); + // double cond = svd.singularValues()(0) / svd.singularValues()(svd.singularValues().size() - 1); + // std::cout << "\n Condition Number from JacobiSVD: " << cond << std::endl; + // + // // Calculate Condition Number from Norms + // double condNumNorms = hqrMatrix.completeOrthogonalDecomposition().pseudoInverse().norm() * hqrMatrix.norm(); + // + // std::cout << "\n Condition Number from Norms: " << condNumNorms << std::endl; } }