| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | #include <iostream> |
| | #include <fstream> |
| | #include <iomanip> |
| |
|
| | #include "main.h" |
| | #include <Eigen/LevenbergMarquardt> |
| | using namespace std; |
| | using namespace Eigen; |
| |
|
| | template<typename Scalar> |
| | struct DenseLM : DenseFunctor<Scalar> |
| | { |
| | typedef DenseFunctor<Scalar> Base; |
| | typedef typename Base::JacobianType JacobianType; |
| | typedef Matrix<Scalar,Dynamic,1> VectorType; |
| | |
| | DenseLM(int n, int m) : DenseFunctor<Scalar>(n,m) |
| | { } |
| | |
| | VectorType model(const VectorType& uv, VectorType& x) |
| | { |
| | VectorType y; |
| | int m = Base::values(); |
| | int n = Base::inputs(); |
| | eigen_assert(uv.size()%2 == 0); |
| | eigen_assert(uv.size() == n); |
| | eigen_assert(x.size() == m); |
| | y.setZero(m); |
| | int half = n/2; |
| | VectorBlock<const VectorType> u(uv, 0, half); |
| | VectorBlock<const VectorType> v(uv, half, half); |
| | for (int j = 0; j < m; j++) |
| | { |
| | for (int i = 0; i < half; i++) |
| | y(j) += u(i)*std::exp(-(x(j)-i)*(x(j)-i)/(v(i)*v(i))); |
| | } |
| | return y; |
| | |
| | } |
| | void initPoints(VectorType& uv_ref, VectorType& x) |
| | { |
| | m_x = x; |
| | m_y = this->model(uv_ref, x); |
| | } |
| | |
| | int operator()(const VectorType& uv, VectorType& fvec) |
| | { |
| | |
| | int m = Base::values(); |
| | int n = Base::inputs(); |
| | eigen_assert(uv.size()%2 == 0); |
| | eigen_assert(uv.size() == n); |
| | eigen_assert(fvec.size() == m); |
| | int half = n/2; |
| | VectorBlock<const VectorType> u(uv, 0, half); |
| | VectorBlock<const VectorType> v(uv, half, half); |
| | for (int j = 0; j < m; j++) |
| | { |
| | fvec(j) = m_y(j); |
| | for (int i = 0; i < half; i++) |
| | { |
| | fvec(j) -= u(i) *std::exp(-(m_x(j)-i)*(m_x(j)-i)/(v(i)*v(i))); |
| | } |
| | } |
| | |
| | return 0; |
| | } |
| | int df(const VectorType& uv, JacobianType& fjac) |
| | { |
| | int m = Base::values(); |
| | int n = Base::inputs(); |
| | eigen_assert(n == uv.size()); |
| | eigen_assert(fjac.rows() == m); |
| | eigen_assert(fjac.cols() == n); |
| | int half = n/2; |
| | VectorBlock<const VectorType> u(uv, 0, half); |
| | VectorBlock<const VectorType> v(uv, half, half); |
| | for (int j = 0; j < m; j++) |
| | { |
| | for (int i = 0; i < half; i++) |
| | { |
| | fjac.coeffRef(j,i) = -std::exp(-(m_x(j)-i)*(m_x(j)-i)/(v(i)*v(i))); |
| | fjac.coeffRef(j,i+half) = -2.*u(i)*(m_x(j)-i)*(m_x(j)-i)/(std::pow(v(i),3)) * std::exp(-(m_x(j)-i)*(m_x(j)-i)/(v(i)*v(i))); |
| | } |
| | } |
| | return 0; |
| | } |
| | VectorType m_x, m_y; |
| | }; |
| |
|
| | template<typename FunctorType, typename VectorType> |
| | int test_minimizeLM(FunctorType& functor, VectorType& uv) |
| | { |
| | LevenbergMarquardt<FunctorType> lm(functor); |
| | LevenbergMarquardtSpace::Status info; |
| | |
| | info = lm.minimize(uv); |
| | |
| | VERIFY_IS_EQUAL(info, 1); |
| | |
| | return info; |
| | } |
| |
|
| | template<typename FunctorType, typename VectorType> |
| | int test_lmder(FunctorType& functor, VectorType& uv) |
| | { |
| | typedef typename VectorType::Scalar Scalar; |
| | LevenbergMarquardtSpace::Status info; |
| | LevenbergMarquardt<FunctorType> lm(functor); |
| | info = lm.lmder1(uv); |
| | |
| | VERIFY_IS_EQUAL(info, 1); |
| | |
| | return info; |
| | } |
| |
|
| | template<typename FunctorType, typename VectorType> |
| | int test_minimizeSteps(FunctorType& functor, VectorType& uv) |
| | { |
| | LevenbergMarquardtSpace::Status info; |
| | LevenbergMarquardt<FunctorType> lm(functor); |
| | info = lm.minimizeInit(uv); |
| | if (info==LevenbergMarquardtSpace::ImproperInputParameters) |
| | return info; |
| | do |
| | { |
| | info = lm.minimizeOneStep(uv); |
| | } while (info==LevenbergMarquardtSpace::Running); |
| | |
| | VERIFY_IS_EQUAL(info, 1); |
| | |
| | return info; |
| | } |
| |
|
| | template<typename T> |
| | void test_denseLM_T() |
| | { |
| | typedef Matrix<T,Dynamic,1> VectorType; |
| | |
| | int inputs = 10; |
| | int values = 1000; |
| | DenseLM<T> dense_gaussian(inputs, values); |
| | VectorType uv(inputs),uv_ref(inputs); |
| | VectorType x(values); |
| | |
| | |
| | uv_ref << -2, 1, 4 ,8, 6, 1.8, 1.2, 1.1, 1.9 , 3; |
| | |
| | |
| | x.setRandom(); |
| | x = 10*x; |
| | x.array() += 10; |
| | dense_gaussian.initPoints(uv_ref, x); |
| | |
| | |
| | VectorBlock<VectorType> u(uv, 0, inputs/2); |
| | VectorBlock<VectorType> v(uv, inputs/2, inputs/2); |
| | |
| | |
| | |
| | |
| | u.setOnes(); v.setOnes(); |
| | test_minimizeLM(dense_gaussian, uv); |
| | |
| | |
| | u.setOnes(); v.setOnes(); |
| | test_lmder(dense_gaussian, uv); |
| | |
| | |
| | v.setOnes(); u.setOnes(); |
| | test_minimizeSteps(dense_gaussian, uv); |
| | |
| | } |
| |
|
| | EIGEN_DECLARE_TEST(denseLM) |
| | { |
| | CALL_SUBTEST_2(test_denseLM_T<double>()); |
| | |
| | |
| | } |
| |
|