#region License Information
/* HeuristicLab
* Copyright (C) 2002-2012 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using HeuristicLab.Collections;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Optimization;
using HeuristicLab.Parameters;
using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
using System;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace HeuristicLab.Analysis {
///
/// A run in which an algorithm is executed for a certain maximum time only.
///
[Item("Iterated Algorithm", "An algorithm that repeats an algorithm until either a certain target value is reached or a maximum budget is exceeded.")]
[Creatable(CreatableAttribute.Categories.TestingAndAnalysis, Priority = 116)]
[StorableClass]
public sealed class IteratedAlgorithm : Algorithm, IStorableContent {
private const string ExecutionTimeResultName = "Execution Time";
private const string BestQualityResultName = "BestQuality";
private const string RandomRestartsResultName = "RandomRestarts";
private const string EvaluatedSolutionsResultName = "EvaluatedSolutions";
private const string EvaluatedMovesResultName = "EvaluatedMoves";
private const string QualityPerClockResultName = "QualityPerClock";
private const string QualityPerEvaluationsResultName = "QualityPerEvaluations";
private const string EvaluationsResultName = "Evaluations";
public string Filename { get; set; }
public override Type ProblemType { get { return typeof(ISingleObjectiveHeuristicOptimizationProblem); } }
public IValueParameter MaximumExecutionTimeParameter {
get { return (IValueParameter)Parameters["MaximumExecutionTime"]; }
}
public IValueParameter MaximumEvaluationsParameter {
get { return (IValueParameter)Parameters["MaximumEvaluations"]; }
}
public IValueParameter TargetQualityParameter {
get { return (IValueParameter)Parameters["TargetQuality"]; }
}
private IFixedValueParameter MoveCostPerSolutionParameter {
get { return (IFixedValueParameter)Parameters["MoveCostPerSolution"]; }
}
private IFixedValueParameter StoreSolutionInRunParameter {
get { return (IFixedValueParameter)Parameters["StoreSolutionInRun"]; }
}
private IValueParameter AlgorithmParameter {
get { return (IValueParameter)Parameters["Algorithm"]; }
}
private IFixedValueParameter AnalyzerParameter {
get { return (IFixedValueParameter)Parameters["Analyzer"]; }
}
[Storable]
private ResultCollection results;
public override ResultCollection Results {
get { return results; }
}
[Storable]
private QualityPerClockAnalyzer perClockAnalyzer;
public QualityPerClockAnalyzer PerClockAnalyzer {
get { return perClockAnalyzer; }
}
[Storable]
private QualityPerEvaluationsAnalyzer perEvaluationsAnalyzer;
public QualityPerEvaluationsAnalyzer PerEvaluationsAnalyzer {
get { return perEvaluationsAnalyzer; }
}
[Storable]
private BestScopeSolutionAnalyzer bestSolutionAnalyzer;
public BestScopeSolutionAnalyzer BestSolutionAnalyzer {
get { return bestSolutionAnalyzer; }
}
public double MoveCostPerSolution {
get { return MoveCostPerSolutionParameter.Value.Value; }
set { MoveCostPerSolutionParameter.Value.Value = value; }
}
public bool StoreSolutionInRun {
get { return StoreSolutionInRunParameter.Value.Value; }
set { StoreSolutionInRunParameter.Value.Value = value; }
}
// algorithm will be set in AfterDeserialization to the value of the parameter
private IAlgorithm algorithm;
public IAlgorithm Algorithm {
get { return algorithm; }
set {
if (algorithm == value) return;
if (algorithm != null) {
DeregisterAlgorithmEvents();
RemoveAlgorithmAnalyzers();
}
algorithm = value;
if (algorithm != null) {
if (algorithm.ExecutionState != ExecutionState.Prepared)
algorithm.Prepare(true);
if (Problem != null)
algorithm.Problem = Problem;
RegisterAlgorithmEvents();
AddAlgorithmAnalyzers();
HandleAlgorithmMaximumEvaluationsChanged();
HandleAlgorithmTargetQualityChanged();
}
if (AlgorithmParameter.Value != algorithm)
AlgorithmParameter.Value = algorithm;
Prepare();
}
}
public MultiAnalyzer Analyzer {
get { return AnalyzerParameter.Value; }
}
private bool Maximization {
get { return Problem != null && ((IValueParameter)((ISingleObjectiveHeuristicOptimizationProblem)Problem).MaximizationParameter).Value.Value; }
}
private bool IsFinished {
get {
var executionTime = Results.ContainsKey(ExecutionTimeResultName) ? ((TimeSpanValue)Results[ExecutionTimeResultName].Value).Value : TimeSpan.Zero;
var evaluations = Results.ContainsKey(EvaluationsResultName) ? ((DoubleValue)Results[EvaluationsResultName].Value).Value : 0;
var bestQuality = Results.ContainsKey(BestQualityResultName) ? ((DoubleValue)Results[BestQualityResultName].Value).Value
: (Maximization ? double.MinValue : double.MaxValue);
var targetValue = TargetQualityParameter.Value != null ? TargetQualityParameter.Value.Value
: Maximization ? double.MaxValue : double.MinValue;
var timeHit = MaximumExecutionTimeParameter.Value != null && executionTime >= MaximumExecutionTimeParameter.Value.Value;
var evalHit = MaximumEvaluationsParameter.Value != null && evaluations >= MaximumEvaluationsParameter.Value.Value;
var targetHit = Maximization && bestQuality >= targetValue || !Maximization && bestQuality <= targetValue;
return forceStop || timeHit || evalHit || targetHit;
}
}
[StorableConstructor]
private IteratedAlgorithm(bool deserializing) : base(deserializing) { }
private IteratedAlgorithm(IteratedAlgorithm original, Cloner cloner)
: base(original, cloner) {
results = cloner.Clone(original.Results);
perClockAnalyzer = cloner.Clone(original.perClockAnalyzer);
perEvaluationsAnalyzer = cloner.Clone(original.perEvaluationsAnalyzer);
bestSolutionAnalyzer = cloner.Clone(original.bestSolutionAnalyzer);
algorithm = cloner.Clone(original.algorithm);
RegisterEventHandlers();
}
public IteratedAlgorithm()
: base() {
results = new ResultCollection();
Parameters.Add(new FixedValueParameter("Analyzer", "Analyzers that should be called in addition to the default algorithm analyzers.", new MultiAnalyzer()));
Parameters.Add(new OptionalValueParameter("MaximumExecutionTime", "The maximum wall-clock time that the algorithm should run."));
Parameters.Add(new OptionalValueParameter("MaximumEvaluations", "The maximum number of function evaluations that the algorithm should run.", new IntValue(100000000)));
Parameters.Add(new OptionalValueParameter("TargetQuality", "The target quality that the algorithm should run for."));
Parameters.Add(new FixedValueParameter("MoveCostPerSolution", "The amount of solution evaluation equivalents of a single move. Use 1 for a black-box scenario.", new DoubleValue(1)));
Parameters.Add(new FixedValueParameter("StoreSolutionInRun", "Whether the solution data types should be kept in the run."));
Parameters.Add(new ValueParameter("Algorithm", "The algorithm to iterate.") { GetsCollected = false }); // due to storage efficiency, by default we don't want to store the algorithm instance in the run
perClockAnalyzer = new QualityPerClockAnalyzer();
perEvaluationsAnalyzer = new QualityPerEvaluationsAnalyzer();
bestSolutionAnalyzer = new BestScopeSolutionAnalyzer();
Analyzer.Operators.Add(perClockAnalyzer, true);
Analyzer.Operators.Add(perEvaluationsAnalyzer, true);
Analyzer.Operators.Add(bestSolutionAnalyzer, StoreSolutionInRun);
RegisterEventHandlers();
}
public override IDeepCloneable Clone(Cloner cloner) {
if (ExecutionState == ExecutionState.Started) throw new InvalidOperationException(string.Format("Clone not allowed in execution state \"{0}\".", ExecutionState));
return new IteratedAlgorithm(this, cloner);
}
[StorableHook(HookType.AfterDeserialization)]
private void AfterDeserialization() {
algorithm = AlgorithmParameter.Value;
// BackwardsCompatibility3.3
#region Backwards compatible code, remove with 3.4
if (Parameters.ContainsKey("TargetValue")) {
var target = ((OptionalValueParameter)Parameters["TargetValue"]).Value;
Parameters.Remove("TargetValue");
Parameters.Add(new OptionalValueParameter("TargetQuality", "The target quality that the algorithm should run for.", target));
}
if (!Parameters.ContainsKey("Analyzer")) {
Parameters.Add(new FixedValueParameter("Analyzer", "Analyzers that should be called in addition to the default algorithm analyzers.", new MultiAnalyzer()));
Analyzer.Operators.Add(perClockAnalyzer, true);
Analyzer.Operators.Add(perEvaluationsAnalyzer, true);
if (Algorithm != null && Algorithm.Parameters.ContainsKey("Analyzer")) {
var analyzerParam = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam != null) {
analyzerParam.Value.Operators.Remove(perClockAnalyzer);
analyzerParam.Value.Operators.Remove(perEvaluationsAnalyzer);
} else {
var analyzerParam2 = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam2 != null) {
analyzerParam2.Value.Operators.Remove(perClockAnalyzer);
analyzerParam2.Value.Operators.Remove(perEvaluationsAnalyzer);
}
}
AddAlgorithmAnalyzers();
}
}
if (bestSolutionAnalyzer == null) {
bestSolutionAnalyzer = new BestScopeSolutionAnalyzer();
Analyzer.Operators.Add(bestSolutionAnalyzer, StoreSolutionInRun);
}
#endregion
RegisterEventHandlers();
}
#region Register Event Handlers
protected override void RegisterProblemEvents() {
var bkParam = ((ISingleObjectiveHeuristicOptimizationProblem)Problem).BestKnownQualityParameter as IValueParameter;
if (bkParam != null) {
bkParam.ValueChanged += Problem_BestKnownQualityParameter_ValueChanged;
}
base.RegisterProblemEvents();
}
protected override void DeregisterProblemEvents() {
base.DeregisterProblemEvents();
var bkParam = ((ISingleObjectiveHeuristicOptimizationProblem)Problem).BestKnownQualityParameter as IValueParameter;
if (bkParam != null) {
bkParam.ValueChanged -= Problem_BestKnownQualityParameter_ValueChanged;
}
}
private void RegisterAlgorithmEvents() {
Algorithm.ExceptionOccurred += Algorithm_ExceptionOccurred;
Algorithm.Paused += Algorithm_Paused;
Algorithm.Stopped += Algorithm_Stopped;
Algorithm.ProblemChanged += Algorithm_ProblemChanged;
#region MaximumEvaluations
IParameter param;
if (Algorithm.Parameters.TryGetValue("MaximumEvaluations", out param)) {
var intFixedValueParam = (param as IFixedValueParameter);
if (intFixedValueParam != null) {
intFixedValueParam.Value.ValueChanged += AlgorithmMaximumEvaluationsOnValueChanged;
} else {
var intValueParam = (param as IValueParameter);
if (intValueParam != null) {
intValueParam.ValueChanged += AlgorithmMaximumEvaluationsParameterOnChanged;
if (intValueParam.Value != null) intValueParam.Value.ValueChanged += AlgorithmMaximumEvaluationsOnValueChanged;
}
}
}
#endregion
#region TargetQuality
if (Algorithm.Parameters.TryGetValue("TargetQuality", out param)) {
var intValueParam = (param as IValueParameter);
if (intValueParam != null) {
intValueParam.ValueChanged += AlgorithmTargetQualityParameterOnChanged;
if (intValueParam.Value != null) intValueParam.Value.ValueChanged += AlgorithmTargetQualityOnValueChanged;
}
}
#endregion
}
private void DeregisterAlgorithmEvents() {
Algorithm.ExceptionOccurred -= Algorithm_ExceptionOccurred;
Algorithm.Paused -= Algorithm_Paused;
Algorithm.Stopped -= Algorithm_Stopped;
Algorithm.ProblemChanged -= Algorithm_ProblemChanged;
#region MaximumEvaluations
IParameter param;
if (Algorithm.Parameters.TryGetValue("MaximumEvaluations", out param)) {
var intFixedValueParam = (param as IFixedValueParameter);
if (intFixedValueParam != null) {
intFixedValueParam.Value.ValueChanged -= AlgorithmMaximumEvaluationsOnValueChanged;
} else {
var intValueParam = (param as IValueParameter);
if (intValueParam != null) {
intValueParam.ValueChanged -= AlgorithmMaximumEvaluationsParameterOnChanged;
if (intValueParam.Value != null) intValueParam.Value.ValueChanged -= AlgorithmMaximumEvaluationsOnValueChanged;
}
}
}
#endregion
#region TargetQuality
if (Algorithm.Parameters.TryGetValue("TargetQuality", out param)) {
var intValueParam = (param as IValueParameter);
if (intValueParam != null) {
intValueParam.ValueChanged -= AlgorithmTargetQualityParameterOnChanged;
if (intValueParam.Value != null) intValueParam.Value.ValueChanged -= AlgorithmTargetQualityOnValueChanged;
}
}
#endregion
}
#endregion
private void RegisterEventHandlers() {
if (Algorithm != null) RegisterAlgorithmEvents();
if (Problem != null) RegisterProblemEvents();
AlgorithmParameter.ValueChanged += AlgorithmParameterOnValueChanged;
StoreSolutionInRunParameter.Value.ValueChanged += StoreSolutionInRunOnValueChanged;
Analyzer.Operators.CollectionReset += AnalyzerOperatorsChanged;
Analyzer.Operators.ItemsAdded += AnalyzerOperatorsChanged;
Analyzer.Operators.ItemsRemoved += AnalyzerOperatorsChanged;
Analyzer.Operators.ItemsReplaced += AnalyzerOperatorsChanged;
MaximumEvaluationsParameter.Value.ValueChanged += MaximumEvaluationsParameterOnValueChanged;
TargetQualityParameter.ValueChanged += TargetQualityParameterChanged;
if (TargetQualityParameter.Value != null) TargetQualityParameter.Value.ValueChanged += TargetQualityParameterOnValueChanged;
}
private bool suppressAnalyzerOperatorEvents = false;
private void AnalyzerOperatorsChanged(object sender, CollectionItemsChangedEventArgs> e) {
if (suppressAnalyzerOperatorEvents) return;
if (!Analyzer.Operators.Contains(perClockAnalyzer)) {
suppressAnalyzerOperatorEvents = true;
try { Analyzer.Operators.Add(perClockAnalyzer, false); } finally { suppressAnalyzerOperatorEvents = false; }
}
if (!Analyzer.Operators.Contains(perEvaluationsAnalyzer)) {
suppressAnalyzerOperatorEvents = true;
try { Analyzer.Operators.Add(perEvaluationsAnalyzer, false); } finally { suppressAnalyzerOperatorEvents = false; }
}
if (!Analyzer.Operators.Contains(bestSolutionAnalyzer)) {
suppressAnalyzerOperatorEvents = true;
try { Analyzer.Operators.Add(bestSolutionAnalyzer, false); } finally { suppressAnalyzerOperatorEvents = false; }
}
}
private void StoreSolutionInRunOnValueChanged(object sender, EventArgs eventArgs) {
Analyzer.Operators.SetItemCheckedState(bestSolutionAnalyzer, StoreSolutionInRun);
}
private void AlgorithmParameterOnValueChanged(object sender, EventArgs eventArgs) {
Algorithm = AlgorithmParameter.Value;
}
#region Prepare, Start, Pause, Stop
public override void Prepare() {
if (Problem == null || Algorithm == null) return;
Algorithm.Prepare(true);
results.Clear();
OnPrepared();
}
public override void Start() {
base.Start();
OnStarted();
Task.Factory.StartNew(Run, null).ContinueWith(t => {
try {
t.Wait();
} catch (AggregateException ex) {
try {
ex.Flatten().Handle(x => x is OperationCanceledException);
} catch (AggregateException remaining) {
if (remaining.InnerExceptions.Count == 1) OnExceptionOccurred(remaining.InnerExceptions[0]);
else OnExceptionOccurred(remaining);
}
}
if (Algorithm.ExecutionState == ExecutionState.Paused) OnPaused();
else OnStopped();
forceStop = false;
});
}
public override void Pause() {
base.Pause();
Algorithm.Pause();
}
private bool forceStop = false;
public override void Stop() {
base.Stop();
if (ExecutionState == ExecutionState.Started) {
forceStop = true;
Algorithm.Stop();
} else if (ExecutionState == ExecutionState.Paused) {
RoundupResults();
Algorithm.Prepare(true);
OnStopped();
}
}
#endregion
private DateTime lastUpdateTime;
private void Run(object state) {
lastUpdateTime = DateTime.UtcNow;
System.Timers.Timer timer = new System.Timers.Timer(250);
timer.AutoReset = true;
timer.Elapsed += timer_Elapsed;
timer.Start();
try {
Run();
} finally {
timer.Elapsed -= timer_Elapsed;
timer.Stop();
ExecutionTime += DateTime.UtcNow - lastUpdateTime;
}
}
private readonly AutoResetEvent algorithmWaitHandle = new AutoResetEvent(false);
private void Run() {
if (!Results.ContainsKey(ExecutionTimeResultName)) Results.Add(new Result(ExecutionTimeResultName, new TimeSpanValue(TimeSpan.Zero)));
if (!Results.ContainsKey(EvaluatedSolutionsResultName)) Results.Add(new Result(EvaluatedSolutionsResultName, new DoubleValue(0)));
if (!Results.ContainsKey(EvaluatedMovesResultName)) Results.Add(new Result(EvaluatedMovesResultName, new DoubleValue(0)));
if (!Results.ContainsKey(EvaluationsResultName)) Results.Add(new Result(EvaluationsResultName, new DoubleValue(0)));
if (!Results.ContainsKey(BestQualityResultName)) Results.Add(new Result(BestQualityResultName, new DoubleValue(double.NaN)));
do {
if (!Results.ContainsKey(RandomRestartsResultName)) Results.Add(new Result(RandomRestartsResultName, new IntValue(0)));
else if (Algorithm.ExecutionState == ExecutionState.Prepared) ((IntValue)Results[RandomRestartsResultName].Value).Value++;
Algorithm.Start();
algorithmWaitHandle.WaitOne();
if (Algorithm.ExecutionState == ExecutionState.Paused) return;
RoundupResults();
Algorithm.Prepare(true);
} while (!IsFinished);
}
private void RoundupResults() {
if (Algorithm == null) return;
var execTime = ((TimeSpanValue)Results[ExecutionTimeResultName].Value);
var solEvals = ((DoubleValue)Results[EvaluatedSolutionsResultName].Value);
var movEvals = ((DoubleValue)Results[EvaluatedMovesResultName].Value);
var restarts = ((IntValue)Results[RandomRestartsResultName].Value);
var evaluations = ((DoubleValue)Results[EvaluationsResultName].Value);
var bestQuality = ((DoubleValue)Results[BestQualityResultName].Value);
var improvement = false;
IResult result;
if (Algorithm.Results.TryGetValue(perEvaluationsAnalyzer.EvaluatedSolutionsParameter.ActualName, out result)) {
var evals = ((IntValue)result.Value).Value;
evaluations.Value += evals;
solEvals.Value += evals;
}
if (Algorithm.Results.TryGetValue(perEvaluationsAnalyzer.EvaluatedMovesParameter.ActualName, out result)) {
var evals = ((IntValue)result.Value).Value;
evaluations.Value += MoveCostPerSolution * evals;
movEvals.Value += evals;
}
if (Algorithm.Results.TryGetValue(perEvaluationsAnalyzer.BestQualityParameter.ActualName, out result)) {
var newBestQuality = ((DoubleValue)result.Value).Value;
if (double.IsNaN(bestQuality.Value)
|| Maximization && newBestQuality > bestQuality.Value
|| !Maximization && newBestQuality < bestQuality.Value) {
bestQuality.Value = newBestQuality;
improvement = true;
}
}
if (Algorithm.Results.TryGetValue(perClockAnalyzer.QualityPerClockParameter.ActualName, out result)) UpdateQualityPerClockResult((IndexedDataTable)result.Value, restarts.Value);
if (Algorithm.Results.TryGetValue(perEvaluationsAnalyzer.QualityPerEvaluationsParameter.ActualName, out result)) UpdateQualityPerEvaluationsResult((IndexedDataTable)result.Value, restarts.Value);
if (StoreSolutionInRun) {
foreach (var r in Algorithm.Results) {
if (r.Name == bestSolutionAnalyzer.BestSolutionResultName || r.Name.ToLower().EndsWith("solution") && improvement) {
if (!Results.TryGetValue(r.Name, out result))
Results.Add(new Result(r.Name, (IItem)r.Value.Clone()));
else result.Value = (IItem)r.Value.Clone();
}
}
}
execTime.Value = ExecutionTime;
}
private void UpdateQualityPerClockResult(IndexedDataTable perClock, int restarts) {
IndexedDataTable dt;
if (!Results.ContainsKey(QualityPerClockResultName)) {
dt = (IndexedDataTable)perClock.Clone();
if (!dt.Rows.ContainsKey("Restarts"))
dt.Rows.Add(new IndexedDataRow("Restarts") {
VisualProperties = {
ChartType = DataRowVisualProperties.DataRowChartType.StepLine,
SecondYAxis = true
}
});
foreach (var v in dt.Rows.First().Values)
dt.Rows["Restarts"].Values.Add(Tuple.Create(v.Item1, 0.0));
Results.Add(new Result(QualityPerClockResultName, dt));
} else {
dt = (IndexedDataTable)Results[QualityPerClockResultName].Value;
var qualityValues = dt.Rows.First().Values;
var restartValues = dt.Rows["Restarts"].Values;
var best = qualityValues.Last().Item2;
var execTime = qualityValues.Last().Item1;
var improvement = false;
foreach (var tupl in perClock.Rows.First().Values) {
if (Maximization && tupl.Item2 > best || !Maximization && tupl.Item2 < best) {
if (!improvement) {
// the last entry always holds the same value, but with highest execution time
qualityValues.RemoveAt(qualityValues.Count - 1);
restartValues.RemoveAt(restartValues.Count - 1);
improvement = true;
}
qualityValues.Add(Tuple.Create(execTime + tupl.Item1, tupl.Item2));
restartValues.Add(Tuple.Create(execTime + tupl.Item1, (double)restarts));
best = tupl.Item2;
}
}
if (!improvement) {
qualityValues.RemoveAt(qualityValues.Count - 1);
restartValues.RemoveAt(restartValues.Count - 1);
}
var totalExecTime = execTime + perClock.Rows.First().Values.Last().Item1;
qualityValues.Add(Tuple.Create(totalExecTime, best));
restartValues.Add(Tuple.Create(totalExecTime, (double)restarts));
}
}
private void UpdateQualityPerEvaluationsResult(IndexedDataTable perEvaluations, int restarts) {
IndexedDataTable dt;
if (!Results.ContainsKey(QualityPerEvaluationsResultName)) {
dt = (IndexedDataTable)perEvaluations.Clone();
if (!dt.Rows.ContainsKey("Restarts"))
dt.Rows.Add(new IndexedDataRow("Restarts") {
VisualProperties = {
ChartType = DataRowVisualProperties.DataRowChartType.StepLine,
SecondYAxis = true
}
});
foreach (var v in dt.Rows.First().Values)
dt.Rows["Restarts"].Values.Add(Tuple.Create(v.Item1, 0.0));
Results.Add(new Result(QualityPerEvaluationsResultName, dt));
} else {
dt = (IndexedDataTable)Results[QualityPerEvaluationsResultName].Value;
var qualityValues = dt.Rows.First().Values;
var restartValues = dt.Rows["Restarts"].Values;
var best = qualityValues.Last().Item2;
var evaluations = qualityValues.Last().Item1;
var improvement = false;
foreach (var tupl in perEvaluations.Rows.First().Values) {
if (Maximization && tupl.Item2 > best || !Maximization && tupl.Item2 < best) {
if (!improvement) {
// the last entry always holds the same value, but with highest evaluations
qualityValues.RemoveAt(qualityValues.Count - 1);
restartValues.RemoveAt(restartValues.Count - 1);
improvement = true;
}
qualityValues.Add(Tuple.Create(evaluations + tupl.Item1, tupl.Item2));
restartValues.Add(Tuple.Create(evaluations + tupl.Item1, (double)restarts));
best = tupl.Item2;
}
}
if (!improvement) {
// add the best quality again as value with highest evaluations
qualityValues.RemoveAt(qualityValues.Count - 1);
restartValues.RemoveAt(restartValues.Count - 1);
}
var totalEvaluations = evaluations + perEvaluations.Rows.First().Values.Last().Item1;
qualityValues.Add(Tuple.Create(totalEvaluations, best));
restartValues.Add(Tuple.Create(totalEvaluations, (double)restarts));
}
}
private void UpdateTargetValueFromBestKnownQuality() {
var bkParam = ((ISingleObjectiveHeuristicOptimizationProblem)Problem).BestKnownQualityParameter as IValueParameter;
if (bkParam != null && bkParam.Value != null)
TargetQualityParameter.Value = new DoubleValue(bkParam.Value.Value);
else if (bkParam != null && bkParam.Value == null)
TargetQualityParameter.Value = null;
}
private void AddAlgorithmAnalyzers() {
if (Algorithm == null) return;
if (!Algorithm.Parameters.ContainsKey("Analyzer")) return;
var analyzerParam = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam != null) {
if (analyzerParam.Value.Operators.Contains(Analyzer)) return;
analyzerParam.Value.Operators.Add(Analyzer, true);
} else {
var analyzerParam2 = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam2 == null) return;
if (analyzerParam2.Value.Operators.Contains(Analyzer)) return;
analyzerParam2.Value.Operators.Add(Analyzer, true);
}
}
private void RemoveAlgorithmAnalyzers() {
if (Algorithm == null) return;
if (!Algorithm.Parameters.ContainsKey("Analyzer")) return;
var analyzerParam = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam != null) {
analyzerParam.Value.Operators.Remove(Analyzer);
} else {
var analyzerParam2 = Algorithm.Parameters["Analyzer"] as IValueParameter;
if (analyzerParam2 != null) {
analyzerParam2.Value.Operators.Remove(Analyzer);
}
}
}
#region Event Handlers
private void Algorithm_ExceptionOccurred(object sender, EventArgs e) {
OnExceptionOccurred(e.Value);
}
private void Algorithm_Paused(object sender, EventArgs e) {
if (ExecutionState == ExecutionState.Paused) return;
algorithmWaitHandle.Set();
}
private void Algorithm_Stopped(object sender, EventArgs e) {
if (ExecutionState == ExecutionState.Paused) return;
algorithmWaitHandle.Set();
}
private void Algorithm_ProblemChanged(object sender, EventArgs e) {
if (Algorithm.Problem != Problem) Problem = (ISingleObjectiveHeuristicOptimizationProblem)Algorithm.Problem;
AddAlgorithmAnalyzers();
}
private void AlgorithmTargetQualityParameterOnChanged(object sender, EventArgs e) {
var doubleValueParam = (sender as IValueParameter);
if (doubleValueParam.Value != null) doubleValueParam.Value.ValueChanged += AlgorithmTargetQualityOnValueChanged;
HandleAlgorithmTargetQualityChanged();
}
private void AlgorithmTargetQualityOnValueChanged(object sender, EventArgs e) {
HandleAlgorithmTargetQualityChanged();
}
private void AlgorithmMaximumEvaluationsParameterOnChanged(object sender, EventArgs eventArgs) {
var intValueParam = (sender as IValueParameter);
if (intValueParam.Value == null) {
intValueParam.Value = new IntValue();
return;
}
intValueParam.Value.ValueChanged += AlgorithmMaximumEvaluationsOnValueChanged;
HandleAlgorithmMaximumEvaluationsChanged();
}
private void AlgorithmMaximumEvaluationsOnValueChanged(object sender, EventArgs eventArgs) {
var intValue = (IntValue)sender;
HandleAlgorithmMaximumEvaluationsChanged();
}
private void Problem_BestKnownQualityParameter_ValueChanged(object sender, EventArgs e) {
var param = sender as IValueParameter;
if (param != null) {
if (param.Value != null) param.Value.ValueChanged += Problem_BestKnownQualityParameter_Value_ValueChanged;
UpdateTargetValueFromBestKnownQuality();
}
}
private void Problem_BestKnownQualityParameter_Value_ValueChanged(object sender, EventArgs e) {
UpdateTargetValueFromBestKnownQuality();
}
protected override void Problem_Reset(object sender, EventArgs eventArgs) {
if (Algorithm != null) AddAlgorithmAnalyzers();
}
protected override void Problem_OperatorsChanged(object sender, EventArgs eventArgs) {
if (Algorithm != null) AddAlgorithmAnalyzers();
}
protected override void OnProblemChanged() {
base.OnProblemChanged();
if (Algorithm != null) Algorithm.Problem = Problem;
if (Problem != null) UpdateTargetValueFromBestKnownQuality();
}
private void MaximumEvaluationsParameterOnValueChanged(object sender, EventArgs eventArgs) {
SynchronizeMaximumEvaluationsParameter();
}
private void TargetQualityParameterChanged(object sender, EventArgs e) {
if (TargetQualityParameter.Value != null) TargetQualityParameter.Value.ValueChanged += TargetQualityParameterOnValueChanged;
HandleAlgorithmTargetQualityChanged();
}
private void TargetQualityParameterOnValueChanged(object sender, EventArgs eventArgs) {
HandleAlgorithmTargetQualityChanged();
}
private void timer_Elapsed(object sender, System.Timers.ElapsedEventArgs e) {
System.Timers.Timer timer = (System.Timers.Timer)sender;
timer.Enabled = false;
DateTime now = DateTime.UtcNow;
ExecutionTime += now - lastUpdateTime;
lastUpdateTime = now;
timer.Enabled = true;
}
#endregion
private void HandleAlgorithmTargetQualityChanged() {
if (Algorithm == null) return;
IParameter param;
if (!Algorithm.Parameters.TryGetValue("TargetQuality", out param)) return;
var doubleValueParam = (param as IValueParameter);
if (doubleValueParam == null) return;
if (doubleValueParam.Value == null && TargetQualityParameter.Value != null) {
doubleValueParam.Value = new DoubleValue(TargetQualityParameter.Value.Value);
} else if (doubleValueParam.Value != null && TargetQualityParameter.Value == null) {
doubleValueParam.Value = null;
} else if (doubleValueParam.Value != null && TargetQualityParameter.Value != null) {
doubleValueParam.Value.Value = TargetQualityParameter.Value.Value;
}
}
private void HandleAlgorithmMaximumEvaluationsChanged() {
if (Algorithm == null) return;
IParameter param;
if (!Algorithm.Parameters.TryGetValue("MaximumEvaluations", out param)) return;
var intValueParam = (param as IValueParameter);
if (intValueParam == null) return;
var value = intValueParam.Value;
if (value.Value > MaximumEvaluationsParameter.Value.Value)
value.Value = MaximumEvaluationsParameter.Value.Value;
}
private void SynchronizeMaximumEvaluationsParameter() {
if (Algorithm == null) return;
IParameter param;
if (!Algorithm.Parameters.TryGetValue("MaximumEvaluations", out param)) return;
var intValueParam = (param as IValueParameter);
if (intValueParam == null) return;
if (intValueParam.Value == null) intValueParam.Value = new IntValue(MaximumEvaluationsParameter.Value.Value);
else intValueParam.Value.Value = MaximumEvaluationsParameter.Value.Value;
}
}
}