[12968] | 1 | #region License Information
|
---|
| 2 | /* HeuristicLab
|
---|
[16565] | 3 | * Copyright (C) 2002-2019 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
|
---|
[12968] | 4 | *
|
---|
| 5 | * This file is part of HeuristicLab.
|
---|
| 6 | *
|
---|
| 7 | * HeuristicLab is free software: you can redistribute it and/or modify
|
---|
| 8 | * it under the terms of the GNU General Public License as published by
|
---|
| 9 | * the Free Software Foundation, either version 3 of the License, or
|
---|
| 10 | * (at your option) any later version.
|
---|
| 11 | *
|
---|
| 12 | * HeuristicLab is distributed in the hope that it will be useful,
|
---|
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
---|
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
---|
| 15 | * GNU General Public License for more details.
|
---|
| 16 | *
|
---|
| 17 | * You should have received a copy of the GNU General Public License
|
---|
| 18 | * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
|
---|
| 19 | */
|
---|
| 20 | #endregion
|
---|
| 21 |
|
---|
| 22 | using System;
|
---|
| 23 | using System.Collections.Generic;
|
---|
| 24 | using HeuristicLab.Common;
|
---|
| 25 | using HeuristicLab.Core;
|
---|
| 26 | using HeuristicLab.Data;
|
---|
| 27 | using HeuristicLab.Operators;
|
---|
| 28 | using HeuristicLab.Parameters;
|
---|
[16565] | 29 | using HEAL.Attic;
|
---|
[12968] | 30 |
|
---|
| 31 | namespace HeuristicLab.Selection {
|
---|
| 32 | [Item("EvolutionStrategyOffspringSelector", "Selects among the offspring population those that are designated successful and discards the unsuccessful offspring, except for some lucky losers. It expects the parent scopes to be below the first sub-scope, and offspring scopes to be below the second sub-scope separated again in two sub-scopes, the first with the failed offspring and the second with successful offspring.")]
|
---|
[16565] | 33 | [StorableType("CE585C3C-5139-44F0-9CB2-CC901A290831")]
|
---|
[12968] | 34 | public class EvolutionStrategyOffspringSelector : SingleSuccessorOperator {
|
---|
[16565] | 35 | [StorableType("F1CA99A7-E9C2-49F3-8030-5CEE407357AA")]
|
---|
[13261] | 36 | private class FitnessComparer : IComparer<IScope> {
|
---|
[12968] | 37 |
|
---|
| 38 | #region IComparer<IScope> Member
|
---|
[16565] | 39 | [Storable]
|
---|
| 40 | private string qualityParameterName;
|
---|
| 41 | [Storable]
|
---|
[13261] | 42 | private bool maximization;
|
---|
[12968] | 43 |
|
---|
[16565] | 44 | [StorableConstructor]
|
---|
| 45 | protected FitnessComparer(StorableConstructorFlag _) { }
|
---|
| 46 |
|
---|
| 47 | public FitnessComparer(string qualityParamName, bool maximization) {
|
---|
[12968] | 48 | this.qualityParameterName = qualityParamName;
|
---|
[13261] | 49 | this.maximization = maximization;
|
---|
[12968] | 50 | }
|
---|
| 51 |
|
---|
[13261] | 52 | // less than zero if x is-better-than y
|
---|
| 53 | // larger than zero if y is-better-than x
|
---|
| 54 | // zero if both are the same
|
---|
[12968] | 55 | public int Compare(IScope x, IScope y) {
|
---|
[13261] | 56 | IVariable quality1, quality2;
|
---|
[12968] | 57 |
|
---|
[13261] | 58 | if (x.Variables.TryGetValue(qualityParameterName, out quality1)
|
---|
| 59 | && y.Variables.TryGetValue(qualityParameterName, out quality2)) {
|
---|
| 60 | DoubleValue left = quality1.Value as DoubleValue;
|
---|
| 61 | DoubleValue right = quality2.Value as DoubleValue;
|
---|
| 62 | var res = left.CompareTo(right);
|
---|
| 63 | if (maximization) return -res; // in the maximization case the largest value should preceed all others in the sort order
|
---|
| 64 | else return res;
|
---|
| 65 | } else
|
---|
| 66 | throw new ArgumentException("Quality variable " + qualityParameterName + " not found.");
|
---|
[12968] | 67 | }
|
---|
| 68 |
|
---|
| 69 | #endregion
|
---|
| 70 | }
|
---|
| 71 |
|
---|
| 72 | public ValueLookupParameter<DoubleValue> MaximumSelectionPressureParameter {
|
---|
| 73 | get { return (ValueLookupParameter<DoubleValue>)Parameters["MaximumSelectionPressure"]; }
|
---|
| 74 | }
|
---|
| 75 | public ValueLookupParameter<DoubleValue> SuccessRatioParameter {
|
---|
| 76 | get { return (ValueLookupParameter<DoubleValue>)Parameters["SuccessRatio"]; }
|
---|
| 77 | }
|
---|
| 78 | public LookupParameter<DoubleValue> SelectionPressureParameter {
|
---|
| 79 | get { return (ValueLookupParameter<DoubleValue>)Parameters["SelectionPressure"]; }
|
---|
| 80 | }
|
---|
| 81 | public LookupParameter<DoubleValue> CurrentSuccessRatioParameter {
|
---|
| 82 | get { return (LookupParameter<DoubleValue>)Parameters["CurrentSuccessRatio"]; }
|
---|
| 83 | }
|
---|
| 84 | public LookupParameter<ItemList<IScope>> OffspringPopulationParameter {
|
---|
| 85 | get { return (LookupParameter<ItemList<IScope>>)Parameters["OffspringPopulation"]; }
|
---|
| 86 | }
|
---|
| 87 | public LookupParameter<ItemList<IScope>> OffspringVirtualPopulationParameter {
|
---|
| 88 | get { return (LookupParameter<ItemList<IScope>>)Parameters["OffspringVirtualPopulation"]; }
|
---|
| 89 | }
|
---|
| 90 | public LookupParameter<IntValue> OffspringPopulationWinnersParameter {
|
---|
| 91 | get { return (LookupParameter<IntValue>)Parameters["OffspringPopulationWinners"]; }
|
---|
| 92 | }
|
---|
| 93 | public ScopeTreeLookupParameter<BoolValue> SuccessfulOffspringParameter {
|
---|
| 94 | get { return (ScopeTreeLookupParameter<BoolValue>)Parameters["SuccessfulOffspring"]; }
|
---|
| 95 | }
|
---|
| 96 | public OperatorParameter OffspringCreatorParameter {
|
---|
| 97 | get { return (OperatorParameter)Parameters["OffspringCreator"]; }
|
---|
| 98 | }
|
---|
| 99 | public LookupParameter<IntValue> EvaluatedSolutionsParameter {
|
---|
| 100 | get { return (LookupParameter<IntValue>)Parameters["EvaluatedSolutions"]; }
|
---|
| 101 | }
|
---|
| 102 | private LookupParameter<IntValue> MaximumEvaluatedSolutionsParameter {
|
---|
| 103 | get { return (LookupParameter<IntValue>)Parameters["MaximumEvaluatedSolutions"]; }
|
---|
| 104 | }
|
---|
| 105 | public ILookupParameter<DoubleValue> QualityParameter {
|
---|
| 106 | get { return (ILookupParameter<DoubleValue>)Parameters["Quality"]; }
|
---|
| 107 | }
|
---|
[13261] | 108 | public ILookupParameter<BoolValue> MaximizationParameter {
|
---|
| 109 | get { return (ILookupParameter<BoolValue>)Parameters["Maximization"]; }
|
---|
| 110 | }
|
---|
[12968] | 111 |
|
---|
| 112 | public IOperator OffspringCreator {
|
---|
| 113 | get { return OffspringCreatorParameter.Value; }
|
---|
| 114 | set { OffspringCreatorParameter.Value = value; }
|
---|
| 115 | }
|
---|
| 116 |
|
---|
| 117 | [StorableConstructor]
|
---|
[16565] | 118 | protected EvolutionStrategyOffspringSelector(StorableConstructorFlag _) : base(_) { }
|
---|
[12968] | 119 | protected EvolutionStrategyOffspringSelector(EvolutionStrategyOffspringSelector original, Cloner cloner) : base(original, cloner) { }
|
---|
| 120 | public override IDeepCloneable Clone(Cloner cloner) {
|
---|
| 121 | return new EvolutionStrategyOffspringSelector(this, cloner);
|
---|
| 122 | }
|
---|
| 123 | public EvolutionStrategyOffspringSelector()
|
---|
| 124 | : base() {
|
---|
| 125 | Parameters.Add(new ValueLookupParameter<DoubleValue>("MaximumSelectionPressure", "The maximum selection pressure which prematurely terminates the offspring selection step."));
|
---|
| 126 | Parameters.Add(new ValueLookupParameter<DoubleValue>("SuccessRatio", "The ratio of successful offspring that has to be produced."));
|
---|
| 127 | Parameters.Add(new ValueLookupParameter<DoubleValue>("SelectionPressure", "The amount of selection pressure currently necessary to fulfill the success ratio."));
|
---|
| 128 | Parameters.Add(new ValueLookupParameter<DoubleValue>("CurrentSuccessRatio", "The current success ratio indicates how much of the successful offspring have already been generated."));
|
---|
| 129 | Parameters.Add(new LookupParameter<ItemList<IScope>>("OffspringPopulation", "Temporary store of the offspring population."));
|
---|
| 130 | Parameters.Add(new LookupParameter<ItemList<IScope>>("OffspringVirtualPopulation", "Temporary store of the offspring population."));
|
---|
| 131 | Parameters.Add(new LookupParameter<IntValue>("OffspringPopulationWinners", "Temporary store the number of successful offspring in the offspring population."));
|
---|
| 132 | Parameters.Add(new ScopeTreeLookupParameter<BoolValue>("SuccessfulOffspring", "True if the offspring was more successful than its parents.", 2));
|
---|
| 133 | Parameters.Add(new OperatorParameter("OffspringCreator", "The operator used to create new offspring."));
|
---|
[13261] | 134 | Parameters.Add(new LookupParameter<IntValue>("EvaluatedSolutions", "The number of solution evaluations."));
|
---|
[12968] | 135 | Parameters.Add(new LookupParameter<IntValue>("MaximumEvaluatedSolutions", "The maximum number of evaluated solutions (approximately)."));
|
---|
| 136 | Parameters.Add(new LookupParameter<DoubleValue>("Quality", "The quality of a child"));
|
---|
[13261] | 137 | Parameters.Add(new LookupParameter<BoolValue>("Maximization", "Flag that indicates if the problem is a maximization problem"));
|
---|
[12968] | 138 | }
|
---|
| 139 |
|
---|
| 140 | public override IOperation Apply() {
|
---|
| 141 | double maxSelPress = MaximumSelectionPressureParameter.ActualValue.Value;
|
---|
| 142 | double successRatio = SuccessRatioParameter.ActualValue.Value;
|
---|
| 143 | IScope scope = ExecutionContext.Scope;
|
---|
| 144 | IScope parents = scope.SubScopes[0];
|
---|
| 145 | IScope offspring = scope.SubScopes[1];
|
---|
| 146 | int populationSize = parents.SubScopes.Count;
|
---|
| 147 |
|
---|
| 148 | // retrieve actual selection pressure and success ratio
|
---|
| 149 | DoubleValue selectionPressure = SelectionPressureParameter.ActualValue;
|
---|
| 150 | if (selectionPressure == null) {
|
---|
| 151 | selectionPressure = new DoubleValue(0);
|
---|
| 152 | SelectionPressureParameter.ActualValue = selectionPressure;
|
---|
| 153 | }
|
---|
| 154 | DoubleValue currentSuccessRatio = CurrentSuccessRatioParameter.ActualValue;
|
---|
| 155 | if (currentSuccessRatio == null) {
|
---|
| 156 | currentSuccessRatio = new DoubleValue(0);
|
---|
| 157 | CurrentSuccessRatioParameter.ActualValue = currentSuccessRatio;
|
---|
| 158 | }
|
---|
| 159 |
|
---|
| 160 | // retrieve next population
|
---|
| 161 | ItemList<IScope> population = OffspringPopulationParameter.ActualValue;
|
---|
| 162 | ItemList<IScope> virtual_population = OffspringVirtualPopulationParameter.ActualValue;
|
---|
| 163 |
|
---|
| 164 | IntValue successfulOffspring;
|
---|
| 165 | if (population == null) {
|
---|
| 166 | population = new ItemList<IScope>();
|
---|
| 167 | OffspringPopulationParameter.ActualValue = population;
|
---|
| 168 | selectionPressure.Value = 0; // initialize selection pressure for this round
|
---|
| 169 | currentSuccessRatio.Value = 0; // initialize current success ratio for this round
|
---|
| 170 | successfulOffspring = new IntValue(0);
|
---|
| 171 | OffspringPopulationWinnersParameter.ActualValue = successfulOffspring;
|
---|
| 172 |
|
---|
| 173 | virtual_population = new ItemList<IScope>();
|
---|
| 174 | OffspringVirtualPopulationParameter.ActualValue = virtual_population;
|
---|
| 175 | } else successfulOffspring = OffspringPopulationWinnersParameter.ActualValue;
|
---|
| 176 |
|
---|
| 177 | int successfulOffspringAdded = 0;
|
---|
| 178 |
|
---|
| 179 | // implement the ActualValue fetch here - otherwise the parent scope would also be included, given that there may be 1000 or more parents, this is quite unnecessary
|
---|
| 180 | string tname = SuccessfulOffspringParameter.TranslatedName;
|
---|
[13261] | 181 | double tmpSelPress = selectionPressure.Value;
|
---|
| 182 | double tmpSelPressInc = 1.0 / populationSize;
|
---|
[12968] | 183 | for (int i = 0; i < offspring.SubScopes.Count; i++) {
|
---|
| 184 | // fetch value
|
---|
| 185 | IVariable tmpVar;
|
---|
| 186 | if (!offspring.SubScopes[i].Variables.TryGetValue(tname, out tmpVar)) throw new InvalidOperationException(Name + ": Could not determine if an offspring was successful or not.");
|
---|
| 187 | BoolValue tmp = (tmpVar.Value as BoolValue);
|
---|
| 188 | if (tmp == null) throw new InvalidOperationException(Name + ": The variable that indicates whether an offspring is successful or not must contain a BoolValue.");
|
---|
| 189 |
|
---|
| 190 | // add to population
|
---|
| 191 | if (tmp.Value) {
|
---|
| 192 | IScope currentOffspring = offspring.SubScopes[i];
|
---|
[13261] | 193 | offspring.SubScopes.RemoveAt(i);
|
---|
| 194 | i--; // next loop should continue with the subscope at index i which replaced currentOffspring
|
---|
[12968] | 195 | population.Add(currentOffspring);
|
---|
| 196 | successfulOffspringAdded++;
|
---|
[13261] | 197 | } else {
|
---|
[12968] | 198 | IScope currentOffspring = offspring.SubScopes[i];
|
---|
[13261] | 199 | offspring.SubScopes.RemoveAt(i);
|
---|
[12968] | 200 | i--;
|
---|
[13261] | 201 | virtual_population.Add(currentOffspring); // add to losers pool
|
---|
[12968] | 202 | }
|
---|
| 203 | tmpSelPress += tmpSelPressInc;
|
---|
| 204 |
|
---|
[13261] | 205 | double tmpSuccessRatio = (successfulOffspring.Value + successfulOffspringAdded) / ((double)populationSize);
|
---|
[12968] | 206 | if (tmpSuccessRatio >= successRatio && (population.Count + virtual_population.Count) >= populationSize)
|
---|
| 207 | break;
|
---|
| 208 | }
|
---|
| 209 | successfulOffspring.Value += successfulOffspringAdded;
|
---|
| 210 |
|
---|
| 211 | // calculate actual selection pressure and success ratio
|
---|
| 212 | selectionPressure.Value = tmpSelPress;
|
---|
| 213 | currentSuccessRatio.Value = successfulOffspring.Value / ((double)populationSize);
|
---|
| 214 |
|
---|
| 215 | // check if enough children have been generated (or limit of selection pressure or evaluted solutions is reached)
|
---|
[13261] | 216 | if (((EvaluatedSolutionsParameter.ActualValue.Value < MaximumEvaluatedSolutionsParameter.ActualValue.Value)
|
---|
| 217 | && (selectionPressure.Value < maxSelPress)
|
---|
| 218 | && (currentSuccessRatio.Value < successRatio))
|
---|
[12968] | 219 | || ((population.Count + virtual_population.Count) < populationSize)) {
|
---|
| 220 | // more children required -> reduce left and start children generation again
|
---|
| 221 | scope.SubScopes.Remove(parents);
|
---|
| 222 | scope.SubScopes.Remove(offspring);
|
---|
| 223 | while (parents.SubScopes.Count > 0) {
|
---|
| 224 | IScope parent = parents.SubScopes[0];
|
---|
[13261] | 225 | parents.SubScopes.RemoveAt(0); // TODO: repeated call of RemoveAt(0) is inefficient?
|
---|
| 226 | scope.SubScopes.Add(parent); // order of subscopes is reversed
|
---|
[12968] | 227 | }
|
---|
| 228 |
|
---|
[13261] | 229 | IOperator offspringCreator = OffspringCreatorParameter.ActualValue as IOperator;
|
---|
| 230 | if (offspringCreator == null) throw new InvalidOperationException(Name + ": More offspring are required, but no operator specified for creating them.");
|
---|
| 231 | return ExecutionContext.CreateOperation(offspringCreator); // this assumes that this operator will be called again indirectly or directly
|
---|
[12968] | 232 | } else {
|
---|
| 233 | // enough children generated
|
---|
[13261] | 234 | var fitnessComparer = new FitnessComparer(QualityParameter.TranslatedName, MaximizationParameter.ActualValue.Value);
|
---|
| 235 | population.Sort(fitnessComparer); // sort individuals by descending fitness
|
---|
[12968] | 236 |
|
---|
[13261] | 237 | // keeps only the best successRatio * populationSize solutions in the population (the remaining ones are added to the virtual population)
|
---|
[12968] | 238 | int removed = 0;
|
---|
| 239 | for (int i = 0; i < population.Count; i++) {
|
---|
| 240 | double tmpSuccessRatio = i / (double)populationSize;
|
---|
| 241 | if (tmpSuccessRatio > successRatio) {
|
---|
[13261] | 242 | virtual_population.Add(population[i]);
|
---|
| 243 | removed++;
|
---|
[12968] | 244 | }
|
---|
| 245 | }
|
---|
| 246 | population.RemoveRange(population.Count - removed, removed);
|
---|
| 247 |
|
---|
| 248 | //fill up population with best remaining children (successful or unsuccessful)
|
---|
[13261] | 249 | virtual_population.Sort(fitnessComparer);
|
---|
[12968] | 250 | int offspringNeeded = populationSize - population.Count;
|
---|
| 251 | for (int i = 0; i < offspringNeeded && i < virtual_population.Count; i++) {
|
---|
[13261] | 252 | population.Add(virtual_population[i]); // children are sorted by descending fitness
|
---|
[12968] | 253 | }
|
---|
| 254 |
|
---|
| 255 | offspring.SubScopes.Clear();
|
---|
| 256 | offspring.SubScopes.AddRange(population);
|
---|
| 257 |
|
---|
| 258 | scope.Variables.Remove(OffspringPopulationParameter.TranslatedName);
|
---|
| 259 | scope.Variables.Remove(OffspringVirtualPopulationParameter.TranslatedName);
|
---|
| 260 | scope.Variables.Remove(OffspringPopulationWinnersParameter.TranslatedName);
|
---|
| 261 | return base.Apply();
|
---|
| 262 | }
|
---|
| 263 | }
|
---|
| 264 | }
|
---|
| 265 | }
|
---|