182 lines
7.2 KiB
C++
182 lines
7.2 KiB
C++
|
/*
|
||
|
* Open BEAGLE
|
||
|
* Copyright (C) 2001-2004 by Christian Gagne and Marc Parizeau
|
||
|
*
|
||
|
* This library is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU Lesser General Public
|
||
|
* License as published by the Free Software Foundation; either
|
||
|
* version 2.1 of the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This library is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
|
* Lesser General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Lesser General Public
|
||
|
* License along with this library; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*
|
||
|
* Contact:
|
||
|
* Laboratoire de Vision et Systemes Numeriques
|
||
|
* Departement de genie electrique et de genie informatique
|
||
|
* Universite Laval, Quebec, Canada, G1K 7P4
|
||
|
* http://vision.gel.ulaval.ca
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*!
|
||
|
* \file beagle/GA/src/MutationCMAFltVecOp.cpp
|
||
|
* \brief Source code of class GA::MutationCMAFltVecOp.
|
||
|
* \author Christian Gagne
|
||
|
* \author Marc Parizeau
|
||
|
* $Revision: 1.8.2.1 $
|
||
|
* $Date: 2007/05/11 19:13:09 $
|
||
|
*/
|
||
|
|
||
|
#include "beagle/GA.hpp"
|
||
|
|
||
|
#include <cfloat>
|
||
|
#include <float.h>
|
||
|
|
||
|
using namespace Beagle;
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Construct CMA-ES mutation operator.
|
||
|
* \param inMutationPbName CMA-ES mutation probability parameter name used in register.
|
||
|
* \param inName Name of the CMA-ES operator.
|
||
|
*/
|
||
|
GA::MutationCMAFltVecOp::MutationCMAFltVecOp(std::string inMutationPbName,
|
||
|
std::string inName) :
|
||
|
MutationOp(inMutationPbName, inName)
|
||
|
{ }
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Register the parameters of the CMA-ES mutation operator.
|
||
|
* \param ioSystem System of the evolution.
|
||
|
*/
|
||
|
void GA::MutationCMAFltVecOp::initialize(System& ioSystem)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
MutationOp::initialize(ioSystem);
|
||
|
if(ioSystem.getRegister().isRegistered("ga.float.maxvalue")) {
|
||
|
mMaxValue = castHandleT<DoubleArray>(ioSystem.getRegister()["ga.float.maxvalue"]);
|
||
|
} else {
|
||
|
std::ostringstream lOSS;
|
||
|
lOSS << "Maximum values assigned to vector's floats. ";
|
||
|
lOSS << "Value can be a scalar, which limit the value for all float ";
|
||
|
lOSS << "vector parameters, or a vector which limit the value for the parameters ";
|
||
|
lOSS << "individually. If the maximum value is smaller than the ";
|
||
|
lOSS << "float vector size, the limit used for the last values of the float vector ";
|
||
|
lOSS << "is equal to the last value of the maximum value vector.";
|
||
|
Register::Description lDescription(
|
||
|
"Maximum vector values",
|
||
|
"DoubleArray",
|
||
|
dbl2str(DBL_MAX),
|
||
|
lOSS.str()
|
||
|
);
|
||
|
mMaxValue = new DoubleArray(1,DBL_MAX);
|
||
|
ioSystem.getRegister().addEntry("ga.float.maxvalue", mMaxValue, lDescription);
|
||
|
}
|
||
|
if(ioSystem.getRegister().isRegistered("ga.float.minvalue")) {
|
||
|
mMinValue = castHandleT<DoubleArray>(ioSystem.getRegister()["ga.float.minvalue"]);
|
||
|
} else {
|
||
|
std::ostringstream lOSS;
|
||
|
lOSS << "Minimum values assigned to vector's floats. ";
|
||
|
lOSS << "Value can be a scalar, which limit the value for all float ";
|
||
|
lOSS << "vector parameters, or a vector which limit the value for the parameters ";
|
||
|
lOSS << "individually. If the minimum value is smaller than the ";
|
||
|
lOSS << "float vector size, the limit used for the last values of the float vector ";
|
||
|
lOSS << "is equal to the last value of the minimum value vector.";
|
||
|
Register::Description lDescription(
|
||
|
"Minimum values",
|
||
|
"DoubleArray",
|
||
|
dbl2str(-DBL_MAX),
|
||
|
lOSS.str()
|
||
|
);
|
||
|
mMinValue = new DoubleArray(1,-DBL_MAX);
|
||
|
ioSystem.getRegister().addEntry("ga.float.minvalue", mMinValue, lDescription);
|
||
|
}
|
||
|
Beagle_StackTraceEndM("void GA::MutationCMAFltVecOp::registerParams(System&)");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Apply CMA-ES mutation on a real-valued GA individual.
|
||
|
* \param ioIndividual Real-valued GA individual to mutate.
|
||
|
* \param ioContext Context of the evolution.
|
||
|
* \return True if the individual is effectively mutated, false if not.
|
||
|
*/
|
||
|
bool GA::MutationCMAFltVecOp::mutate(Individual& ioIndividual, Context& ioContext)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
Beagle_LogVerboseM(
|
||
|
ioContext.getSystem().getLogger(),
|
||
|
"mutation", "Beagle::GA::MutationCMAFltVecOp",
|
||
|
std::string("Applying CMA-ES mutation")
|
||
|
);
|
||
|
|
||
|
if(ioIndividual.size()!=1) throw Beagle_RunTimeExceptionM(
|
||
|
"CMA-ES mutation can't be applied on individuals composed by more than one float vector");
|
||
|
|
||
|
// Get the appropriate CMA values from the CMA holder component.
|
||
|
Component::Handle lHolderComponent = ioContext.getSystem().getComponent("CMAHolder");
|
||
|
if(lHolderComponent==NULL)
|
||
|
throw Beagle_RunTimeExceptionM("No CMA holder component found in the system!");
|
||
|
GA::CMAHolder::Handle lCMAHolder = castHandleT<GA::CMAHolder>(lHolderComponent);
|
||
|
if(lCMAHolder==NULL)
|
||
|
throw Beagle_RunTimeExceptionM("Component named \"CMAHolder\" found is not of the good type!");
|
||
|
GA::CMAHolder::iterator lIterVal = lCMAHolder->find(ioContext.getDemeIndex());
|
||
|
if(lIterVal == lCMAHolder->end()) {
|
||
|
std::ostringstream lOSS;
|
||
|
lOSS << "CMA values of index \"" << ioContext.getDemeIndex() << "\" not found in CMA holder ";
|
||
|
lOSS << "component. Consider having a breeding pipeline with an appropriate CMA replacement ";
|
||
|
lOSS << "strategy that would correctly initialize the CMA values at the given index.";
|
||
|
throw Beagle_RunTimeExceptionM(lOSS.str());
|
||
|
}
|
||
|
GA::CMAValues& lValues = lIterVal->second;
|
||
|
|
||
|
GA::FloatVector::Handle lVector=castHandleT<GA::FloatVector>(ioIndividual[0]);
|
||
|
Beagle_AssertM(lVector->size()==lValues.mD.size());
|
||
|
const double lSigma=lValues.mSigma.getWrappedValue();
|
||
|
Vector lArz(lVector->size());
|
||
|
for(unsigned int i=0; i<lArz.size(); ++i) {
|
||
|
double lStdErr = lSigma * lValues.mD[i];
|
||
|
if(isNaN(lStdErr) || (lStdErr<0.)) {
|
||
|
lArz[i] = 0.0;
|
||
|
std::ostringstream lOSS;
|
||
|
lOSS << "Warning: standard deviation for CMA-ES mutation over the " << uint2ordinal(i+1);
|
||
|
lOSS << " principal component is NaN or negative (" << lStdErr << "). ";
|
||
|
lOSS << "Mutation of this component has been cancelled.";
|
||
|
Beagle_LogBasicM(
|
||
|
ioContext.getSystem().getLogger(),
|
||
|
"mutation", "Beagle::GA::MutationCMAFltVecOp",
|
||
|
lOSS.str()
|
||
|
);
|
||
|
}
|
||
|
else lArz[i] = ioContext.getSystem().getRandomizer().rollGaussian(0.0, lStdErr);
|
||
|
}
|
||
|
Vector lMutFactor;
|
||
|
lValues.mB.multiply((Matrix&)lMutFactor, lArz);
|
||
|
Beagle_AssertM(lMutFactor.size()==lVector->size());
|
||
|
for(unsigned int i=0; i<lVector->size(); ++i) {
|
||
|
const double lMaxVal = i<mMaxValue->size() ? (*mMaxValue)[i] : mMaxValue->back();
|
||
|
const double lMinVal = i<mMinValue->size() ? (*mMinValue)[i] : mMinValue->back();
|
||
|
(*lVector)[i] += lMutFactor[i];
|
||
|
if((*lVector)[i] > lMaxVal) (*lVector)[i] = lMaxVal;
|
||
|
if((*lVector)[i] < lMinVal) (*lVector)[i] = lMinVal;
|
||
|
}
|
||
|
|
||
|
Beagle_LogDebugM(
|
||
|
ioContext.getSystem().getLogger(),
|
||
|
"mutation", "Beagle::GA::MutationCMAFltVecOp",
|
||
|
std::string("Float vector generated by CMA-ES mutation: ")+lVector->serialize()
|
||
|
);
|
||
|
return true;
|
||
|
Beagle_StackTraceEndM("bool GA::MutationCMAFltVecOp::mutate(Individual& ioIndividual, Context& ioContext)");
|
||
|
}
|
||
|
|
||
|
|
||
|
|