⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 gagenome.h

📁 遗传算法工具箱C++
💻 H
字号:
// $Header$/* ----------------------------------------------------------------------------  genome.h  mbwall 28jul94  Copyright (c) 1995 Massachusetts Institute of Technology DESCRIPTION:  The base genome class just defines the genome interface - how to mutate,crossover, evaluate, etc.  When you create your own genome, multiply inheritfrom the base genome class and the data type that you want to use.  Use thedata type to store the information and use the genome part to tell the GA howit should operate on the data.  See comments below for further details.---------------------------------------------------------------------------- */#ifndef _ga_genome_h_#define _ga_genome_h_#include "gaid.h"#include "gaconfig.h"#include "gaerror.h"#include "GAEvalData.h"#include "std_stream.h"class GAGeneticAlgorithm;class GAGenome;/* ----------------------------------------------------------------------------Genome-------------------------------------------------------------------------------Deriving your own genomes:  For any derived class be sure to define the canonical methods:  constructor,copy constructor, operator=, and destructor.  Make sure that you check for aself-copy in your copy method (it is possible that a genome will beselected to cross with itself, and self-copying is not out of the question)  To work properly with the GAlib, you MUST define the following:       YourGenome( -default-args-for-your-genome )       YourGenome(const YourGenome&)       virtual ~YourGenome()       virtual GAGenome* clone(GAGenome::CloneMethod)       virtual copy(const GAGenome&)  If your genome class defines any new properties you should to define:       virtual int read(istream&)       virtual int write(ostream&) const       virtual int equal(const GAGenome&) const      When you derive a genome, don't forget to use the _evaluated flag to   indicate when the state of the genome has changed and an evaluation is   needed.    Assign a default crossover method so that users don't have to assign one  unless they want to.  Do this in the constructor.      It is a good idea to define an identity for your genome (especially if   you will be using it in an environment with multiple genome types running   around).  Use the DefineIdentity/DeclareIdentity macros (defined in id.h)   to do this in your class definition.Brief overview of the member functions:initialize  Use this method to set the initial state of your genomes once they have  been created.  This initialization is for setting up the genome's state,  not for doing the basic mechanics of genome class management.  The   default behaviour of this method is to change randomly the contents of the   genome.  If you want to bias your initial population, this is where to  make that happen.     The initializer is used to initialize the genome (duh).  Notice that the  state of the genome is unknown - memory may or may not have been allocated,   and the genome may or may not have been used before.  So your initializer   should first clean up as needed, then do its thing.  The initializer may be   called any number of times (unlike a class constructor which is called only   once for a given instance). mutate  Mutate the genome with probability as specified.  What mutation means  depends upon the data type of the genome.  For example, you could have   a bit string in which 50% mutation means each bit has a 50% chance of  getting flipped, or you could have a tree in which 50% mutation means each  node has a 50% chance of getting deleted, or you could have a bit string  in which 50% mutation means 50% of the bits ACTUALLY get flipped.    The mutations member returns the number of mutations since the genome   was initialized.    The mutator makes a change to the genome with likeliehood determined by the  mutation rate parameter.  The exact meaning of mutation is up to you, as is   the specific meaning of the mutation rate.  The function returns the number   of mutations that actually occurred.crossover  Genomes don't really have any clue about other genomes, so we don't make  the crossover a member function.  Instead, each genome kind of knows how  to mate with other genomes to generate offspring, but they are not  capable of doing it themselves.  The crossover member function is used to  set the default mating mode for the genomes - it does not actually perform  the crossover.  This way the GA can use asexual crossover if it wants to  (but genomes only know how to do the default sexual crossover).    This also lets you do funky stuff like crossover between different data  types and group sex to generate new offspring.     We define two types of crossover:  sexual and asexual.  Most GAlib  algorithms use the sexual crossover, but both are available.  Each genome   knows the preferred crossover method, but none is capable of reproducing.    The genetic algorithm must actually perform the mating because it involves   another genome (as parent and/or child).evaluator  Set the genome's objective function.  This also sets marks the evaluated  flag to indicate that the genome must be re-evaluated.    Evaluation happens on-demand - the objective score is not calculated until  it is requested.  Then it is cached so that it does not need to be re-  calculated each time it is requested.  This means that any member function  that modifies the state of the genome must also set the evaluated flag to  indicate that the score must be recalculated.    The genome objective function is used by the GA to evaluate each member of  the population.comparator  This method is used to determine how similar two genomes are.  If you want   to use a different comparison method without deriving a new class, then use   the comparator function to do so.  For example, you may want to do phenotype-  based comparisons rather than genotype-based comparisons.    In many cases we have to compare two genomes to determine how similar or  different they are.  In traditional GA literature this type of function is   referred to as a 'distance' function, probably because bit strings can be  compared using the Hamming distance as a measure of similarity.  In GAlib, we  define a genome comparator function that does exactly this kind of   comparison.    If the genomes are identical, the similarity function should return a   value of 0.0, if completely different then return a value greater than 0.    The specific definition of what "the same" and what "different" mean is up   to you.  Most of the default comparators use the genotype for the comparison,  but you can use the phenotype if you prefer.  There is no upper limit to the  distance score as far as GAlib is concerned.    The no-op function returns a -1 to signify that the comparison failed.evalData  The evalData member is useful if you do not want to derive a new genome class  but want to store data with each genome.  When you clone a genome, the eval  data also gets cloned so that each genome has its own eval data (unlike the  user data pointer described next which is shared by all genomes).userData  The userData member is used to provide all genomes access to the same user  data.  This can be a pointer to anything you want.  Any genome cloned from  another will share the same userData as the original.  This means that all  of the genomes in a population, for example, share the same userData.score  Evaluate the 'performance' of the genome using the objective function.  The score is kept in the 'score' member.  The 'evaluated' member tells us  whether or not we can trust the score.  Be sure to set/unset the 'evaluated'  member as appropriate (eg cross and mutate change the contents of the   genome so they unset the 'evaluated' flag).    If there is no objective function, then simply return the score.  This   allows us to use population-based evaluation methods (where the population  method sets the score of each genome).clone  This method allocates space for a new genome and copies the original into  the new space.  Depending on the argument, it either copies the entire   original or just parts of the original.  For some data types, clone contents  and clone attributes will do the same thing.  If your data type requires  significant overhead for initialization, then you'll probably want to   distinguish between cloning contents and cloning attributes.clone(cont)  Clone the contents of the genome.  Returns a pointer to a GAGenome  (which actually points to a genome of the type that was cloned).  This is  a 'deep copy' in which every part of the genome is duplicated.clone(attr)  Clone the attributes of the genome.  This method does nothing to the  contents of the genome.  It does NOT call the initialization method.  For  some data types this is the same thing as cloning the contents.---------------------------------------------------------------------------- */class GAGenome : public GAID {public:  GADefineIdentity("GAGenome", GAID::Genome);public:  typedef float (*Evaluator)(GAGenome &);  typedef void  (*Initializer)(GAGenome &);  typedef int   (*Mutator)(GAGenome &, float);  typedef float (*Comparator)(const GAGenome&, const GAGenome&);  typedef int   (*SexualCrossover)(const GAGenome&, const GAGenome&, 				   GAGenome*, GAGenome*);  typedef int   (*AsexualCrossover)(const GAGenome&, GAGenome*);  public:  static void  NoInitializer(GAGenome &);  static int   NoMutator(GAGenome &, float);  static float NoComparator(const GAGenome&, const GAGenome&);public:  enum Dimension {LENGTH = 0, WIDTH = 0, HEIGHT = 1, DEPTH = 2};  enum CloneMethod {CONTENTS = 0, ATTRIBUTES = 1};  enum {FIXED_SIZE = -1, ANY_SIZE = -10};public:// The GNU compiler sucks.  It won't recognize No*** as a member of the genome// class.  So we have to use 0 as the defaults then check in the constructor.  GAGenome(Initializer i=0, Mutator m=0, Comparator c=0);  GAGenome(const GAGenome&);  GAGenome & operator=(const GAGenome & arg){copy(arg); return *this;}  virtual ~GAGenome();  virtual GAGenome* clone(CloneMethod flag=CONTENTS) const;  virtual void copy(const GAGenome &);#ifdef GALIB_USE_STREAMS  virtual int read(STD_ISTREAM &)    { GAErr(GA_LOC, className(), "read", gaErrOpUndef); return 0; }  virtual int write(STD_OSTREAM &) const    { GAErr(GA_LOC, className(), "write", gaErrOpUndef); return 0; }#endif  virtual int equal(const GAGenome &) const    { GAErr(GA_LOC, className(), "equal", gaErrOpUndef); return 1; }  virtual int notequal(const GAGenome & g) const    { return (equal(g) ? 0 : 1); }public:  int nevals() const {return _neval;}  float score() const { evaluate(); return _score; }  float score(float s){ _evaluated=gaTrue; return _score=s; }  float fitness(){return _fitness;}  float fitness(float f){return _fitness = f;}  GAGeneticAlgorithm* geneticAlgorithm() const {return ga;}  GAGeneticAlgorithm* geneticAlgorithm(GAGeneticAlgorithm& g){return(ga=&g);}  void * userData() const {return ud;}  void * userData(void * u){return(ud=u);}  GAEvalData * evalData() const {return evd;}  GAEvalData * evalData(const GAEvalData& o)    {delete evd; evd = o.clone(); return evd;}  float evaluate(GABoolean flag = gaFalse) const;  Evaluator evaluator() const {return eval;}  Evaluator evaluator(Evaluator f) { _evaluated=gaFalse; return(eval=f); }  void initialize(){_evaluated=gaFalse; _neval=0; (*init)(*this);}  Initializer initializer() const {return init;}  Initializer initializer(Initializer op){return (init=op);}  int mutate(float p){ return((*mutr)(*this,p)); }  Mutator mutator() const {return mutr;}  Mutator mutator(Mutator op){return (mutr=op);}  float compare(const GAGenome& g) const {return (*cmp)(*this,g);}  Comparator comparator() const {return cmp;}  Comparator comparator(Comparator c){return (cmp=c);}  SexualCrossover crossover(SexualCrossover f) { return sexcross = f; }  SexualCrossover sexual() const { return sexcross; }  AsexualCrossover crossover(AsexualCrossover f) { return asexcross = f; }  AsexualCrossover asexual() const { return asexcross; }protected:  float _score;			// value returned by the objective function  float _fitness;		// (possibly scaled) fitness score  GABoolean _evaluated;		// has this genome been evaluated?  unsigned int _neval;		// how many evaluations since initialization?  GAGeneticAlgorithm* ga;	// the ga that is using this genome  void* ud;		        // pointer to user data  Evaluator eval;		// objective function  GAEvalData* evd;		// evaluation data (specific to each genome)  Mutator mutr;			// the mutation operator to use for mutations  Initializer init;		// how to initialize this genome  Comparator cmp;		// how to compare two genomes of this type  SexualCrossover sexcross;	// preferred sexual mating method  AsexualCrossover asexcross;	// preferred asexual mating method};#ifdef GALIB_USE_STREAMSinline STD_OSTREAM & operator<< (STD_OSTREAM& os, const GAGenome& genome){ genome.write(os); return(os); }inline STD_ISTREAM & operator>> (STD_ISTREAM & is, GAGenome& genome){ genome.read(is); return(is); }#endifinline int operator== (const GAGenome& a, const GAGenome& b){ return a.equal(b); }inline int operator!= (const GAGenome& a, const GAGenome& b){ return a.notequal(b); }#endif

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -