#region License Information
/* HeuristicLab
* Copyright (C) 2002-2019 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 System.Collections.Generic;
using System.Linq;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Encodings.RealVectorEncoding;
using HeuristicLab.Operators;
using HeuristicLab.Optimization;
using HeuristicLab.Parameters;
using HEAL.Attic;
namespace HeuristicLab.Problems.ParameterOptimization {
[Item("BestSolutionsAnalyzer", "Tracks the best parameter vector solutions of the current algorithm run.")]
[StorableType("4882160E-6022-4AFC-AD84-9D7D7FF55562")]
public class BestSolutionsAnalyzer : SingleSuccessorOperator, IAnalyzer {
private const string MaximizationParameterName = "Maximization";
private const string ParameterVectorParameterName = "RealVector";
private const string ParameterNamesParameterName = "ParameterNames";
private const string QualityParameterName = "Quality";
private const string PreviousBestQualityParameterName = "PreviousBestQuality";
private const string BestQualityParameterName = "BestQuality";
private const string BestKnownQualityParameterName = "BestKnownQuality";
private const string ResultsParameterName = "Results";
private const string BestSolutionsResultName = "Best Solutions Store";
public virtual bool EnabledByDefault {
get { return false; }
}
public ILookupParameter MaximizationParameter {
get { return (ILookupParameter)Parameters[MaximizationParameterName]; }
}
public IScopeTreeLookupParameter ParameterVectorParameter {
get { return (IScopeTreeLookupParameter)Parameters[ParameterVectorParameterName]; }
}
public ILookupParameter ParameterNamesParameter {
get { return (ILookupParameter)Parameters[ParameterNamesParameterName]; }
}
public IScopeTreeLookupParameter QualityParameter {
get { return (IScopeTreeLookupParameter)Parameters[QualityParameterName]; }
}
public ILookupParameter PreviousBestQualityParameter {
get { return (ILookupParameter)Parameters[PreviousBestQualityParameterName]; }
}
public ILookupParameter BestQualityParameter {
get { return (ILookupParameter)Parameters[BestQualityParameterName]; }
}
public ILookupParameter BestKnownQualityParameter {
get { return (ILookupParameter)Parameters[BestKnownQualityParameterName]; }
}
public IValueLookupParameter ResultsParameter {
get { return (IValueLookupParameter)Parameters[ResultsParameterName]; }
}
[StorableConstructor]
protected BestSolutionsAnalyzer(StorableConstructorFlag _) : base(_) { }
protected BestSolutionsAnalyzer(BestSolutionsAnalyzer original, Cloner cloner)
: base(original, cloner) { }
public override IDeepCloneable Clone(Cloner cloner) {
return new BestSolutionsAnalyzer(this, cloner);
}
public BestSolutionsAnalyzer()
: base() {
Parameters.Add(new LookupParameter(MaximizationParameterName, "True if the problem is a maximization problem."));
Parameters.Add(new ScopeTreeLookupParameter(ParameterVectorParameterName, "The parameter vector which should be evaluated."));
Parameters.Add(new LookupParameter(ParameterNamesParameterName, "The names of the elements in the parameter vector."));
Parameters.Add(new ScopeTreeLookupParameter(QualityParameterName, "The quality name for the parameter vectors."));
Parameters.Add(new LookupParameter(PreviousBestQualityParameterName, "The best quality of the previous iteration."));
Parameters.Add(new LookupParameter(BestQualityParameterName, "The best quality found so far."));
Parameters.Add(new LookupParameter(BestKnownQualityParameterName, "The quality of the best known solution."));
Parameters.Add(new ValueLookupParameter(ResultsParameterName, "The result collection where the results should be stored."));
}
public override IOperation Apply() {
ItemArray parameterVectors = ParameterVectorParameter.ActualValue;
ItemArray qualities = QualityParameter.ActualValue;
bool max = MaximizationParameter.ActualValue.Value;
DoubleValue bestKnownQuality = BestKnownQualityParameter.ActualValue;
var solutions = parameterVectors.Zip(qualities, (ParameterVector, Quality) => new { ParameterVector, Quality });
if (max) solutions = solutions.MaxItems(s => s.Quality.Value);
else solutions = solutions.MinItems(s => s.Quality.Value);
if (BestQualityParameter.ActualValue == null) {
if (max) BestQualityParameter.ActualValue = new DoubleValue(double.MinValue);
else BestQualityParameter.ActualValue = new DoubleValue(double.MaxValue);
}
if (PreviousBestQualityParameter.ActualValue == null)
PreviousBestQualityParameter.ActualValue = (DoubleValue)BestQualityParameter.ActualValue.Clone();
//add result for best solutions
ResultCollection results = ResultsParameter.ActualValue;
if (!results.ContainsKey(BestSolutionsResultName))
results.Add(new Result(BestSolutionsResultName, new ItemSet(new DoubleArrayEqualityComparer())));
var previousBestQuality = PreviousBestQualityParameter.ActualValue.Value;
var bestQuality = solutions.First().Quality.Value;
var bestSolutions = (ItemSet)results[BestSolutionsResultName].Value;
//clear best solutions if new found quality is better than the existing one
if (max && bestQuality > previousBestQuality || !max && bestQuality < previousBestQuality)
bestSolutions.Clear();
//add new found solutions
if (max && bestQuality >= BestQualityParameter.ActualValue.Value
|| !max && bestQuality <= BestQualityParameter.ActualValue.Value) {
foreach (var solution in solutions) {
var newSolution = (DoubleArray)solution.ParameterVector.Clone();
newSolution.ElementNames = ParameterNamesParameter.ActualValue;
bestSolutions.Add(newSolution);
}
}
//update best quality
if (max && bestQuality >= BestQualityParameter.ActualValue.Value
|| !max && bestQuality <= BestQualityParameter.ActualValue.Value) {
BestQualityParameter.ActualValue.Value = bestQuality;
}
//update best known quality
if (bestKnownQuality == null || max && bestQuality > bestKnownQuality.Value
|| !max && bestQuality < bestKnownQuality.Value) {
BestKnownQualityParameter.ActualValue = new DoubleValue(bestQuality);
}
PreviousBestQualityParameter.ActualValue = (DoubleValue)BestQualityParameter.ActualValue.Clone();
return base.Apply();
}
}
[StorableType("EB92A47A-F96B-4C42-9D31-EF4992320794")]
public class DoubleArrayEqualityComparer : IEqualityComparer {
public bool Equals(DoubleArray x, DoubleArray y) {
if (x == null && y == null) return true;
if (x == null) return false;
if (y == null) return false;
return x.SequenceEqual(y);
}
public int GetHashCode(DoubleArray obj) {
if (obj == null) return 0;
return (int)obj.Aggregate(23L, (current, item) => current ^ System.BitConverter.DoubleToInt64Bits(item));
}
}
}