Path: blob/master/modules/core/test/test_conjugate_gradient.cpp
16337 views
/*M///////////////////////////////////////////////////////////////////////////////////////1//2// IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.3//4// By downloading, copying, installing or using the software you agree to this license.5// If you do not agree to this license, do not download, install,6// copy or use the software.7//8//9// License Agreement10// For Open Source Computer Vision Library11//12// Copyright (C) 2013, OpenCV Foundation, all rights reserved.13// Third party copyrights are property of their respective owners.14//15// Redistribution and use in source and binary forms, with or without modification,16// are permitted provided that the following conditions are met:17//18// * Redistribution's of source code must retain the above copyright notice,19// this list of conditions and the following disclaimer.20//21// * Redistribution's in binary form must reproduce the above copyright notice,22// this list of conditions and the following disclaimer in the documentation23// and/or other materials provided with the distribution.24//25// * The name of the copyright holders may not be used to endorse or promote products26// derived from this software without specific prior written permission.27//28// This software is provided by the copyright holders and contributors "as is" and29// any express or implied warranties, including, but not limited to, the implied30// warranties of merchantability and fitness for a particular purpose are disclaimed.31// In no event shall the OpenCV Foundation or contributors be liable for any direct,32// indirect, incidental, special, exemplary, or consequential damages33// (including, but not limited to, procurement of substitute goods or services;34// loss of use, data, or profits; or business interruption) however caused35// and on any theory of liability, whether in contract, strict liability,36// or tort (including negligence or otherwise) arising in any way out of37// the use of this software, even if advised of the possibility of such damage.38//39//M*/40#include "test_precomp.hpp"4142namespace opencv_test { namespace {4344static void mytest(cv::Ptr<cv::ConjGradSolver> solver,cv::Ptr<cv::MinProblemSolver::Function> ptr_F,cv::Mat& x,45cv::Mat& etalon_x,double etalon_res){46solver->setFunction(ptr_F);47//int ndim=MAX(step.cols,step.rows);48double res=solver->minimize(x);49std::cout<<"res:\n\t"<<res<<std::endl;50std::cout<<"x:\n\t"<<x<<std::endl;51std::cout<<"etalon_res:\n\t"<<etalon_res<<std::endl;52std::cout<<"etalon_x:\n\t"<<etalon_x<<std::endl;53double tol = 1e-2;54ASSERT_TRUE(std::abs(res-etalon_res)<tol);55/*for(cv::Mat_<double>::iterator it1=x.begin<double>(),it2=etalon_x.begin<double>();it1!=x.end<double>();it1++,it2++){56ASSERT_TRUE(std::abs((*it1)-(*it2))<tol);57}*/58std::cout<<"--------------------------\n";59}6061class SphereF_CG:public cv::MinProblemSolver::Function{62public:63int getDims() const { return 4; }64double calc(const double* x)const{65return x[0]*x[0]+x[1]*x[1]+x[2]*x[2]+x[3]*x[3];66}67// use automatically computed gradient68/*void getGradient(const double* x,double* grad){69for(int i=0;i<4;i++){70grad[i]=2*x[i];71}72}*/73};74class RosenbrockF_CG:public cv::MinProblemSolver::Function{75int getDims() const { return 2; }76double calc(const double* x)const{77return 100*(x[1]-x[0]*x[0])*(x[1]-x[0]*x[0])+(1-x[0])*(1-x[0]);78}79void getGradient(const double* x,double* grad){80grad[0]=-2*(1-x[0])-400*(x[1]-x[0]*x[0])*x[0];81grad[1]=200*(x[1]-x[0]*x[0]);82}83};8485TEST(Core_ConjGradSolver, regression_basic){86cv::Ptr<cv::ConjGradSolver> solver=cv::ConjGradSolver::create();87#if 188{89cv::Ptr<cv::MinProblemSolver::Function> ptr_F(new SphereF_CG());90cv::Mat x=(cv::Mat_<double>(4,1)<<50.0,10.0,1.0,-10.0),91etalon_x=(cv::Mat_<double>(1,4)<<0.0,0.0,0.0,0.0);92double etalon_res=0.0;93mytest(solver,ptr_F,x,etalon_x,etalon_res);94}95#endif96#if 197{98cv::Ptr<cv::MinProblemSolver::Function> ptr_F(new RosenbrockF_CG());99cv::Mat x=(cv::Mat_<double>(2,1)<<0.0,0.0),100etalon_x=(cv::Mat_<double>(2,1)<<1.0,1.0);101double etalon_res=0.0;102mytest(solver,ptr_F,x,etalon_x,etalon_res);103}104#endif105}106107}} // namespace108109110