#region License Information
/* HeuristicLab
* Copyright (C) 2002-2017 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;
using System.Linq;
using HeuristicLab.Analysis;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Operators;
using HeuristicLab.Optimization;
using HeuristicLab.Optimization.Operators;
using HeuristicLab.Parameters;
using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
using HeuristicLab.PluginInfrastructure;
using HeuristicLab.Random;
namespace HeuristicLab.Algorithms.GRASP {
///
/// The algorithm combines the Greedy Randomized Adaptive Search Procedure (GRASP) with Path Relinking and is described in Mateus, G., Resende, M., and Silva, R. 2011. GRASP with path-relinking for the generalized quadratic assignment problem. Journal of Heuristics 17, Springer Netherlands, pp. 527-565.
///
[Item("GRASP+PR", "The algorithm combines the Greedy Randomized Adaptive Search Procedure (GRASP) with Path Relinking and is described in Mateus, G., Resende, M., and Silva, R. 2011. GRASP with path-relinking for the generalized quadratic assignment problem. Journal of Heuristics 17, Springer Netherlands, pp. 527-565.")]
[Creatable(CreatableAttribute.Categories.PopulationBasedAlgorithms)]
[StorableClass]
public sealed class GRASPWithPathRelinking : HeuristicOptimizationEngineAlgorithm, IStorableContent {
public string Filename { get; set; }
#region Problem Properties
public override Type ProblemType {
get { return typeof(ISingleObjectiveHeuristicOptimizationProblem); }
}
public new ISingleObjectiveHeuristicOptimizationProblem Problem {
get { return (ISingleObjectiveHeuristicOptimizationProblem)base.Problem; }
set { base.Problem = value; }
}
#endregion
#region Parameter Properties
private IValueParameter SetSeedRandomlyParameter {
get { return (IValueParameter)Parameters["SetSeedRandomly"]; }
}
private IValueParameter SeedParameter {
get { return (IValueParameter)Parameters["Seed"]; }
}
private IValueParameter AnalyzerParameter {
get { return (IValueParameter)Parameters["Analyzer"]; }
}
public IValueParameter MaximumIterationsParameter {
get { return (IValueParameter)Parameters["MaximumIterations"]; }
}
private IFixedValueParameter EliteSetSizeParameter {
get { return (IFixedValueParameter)Parameters["EliteSetSize"]; }
}
private IFixedValueParameter LocalImprovementMaximumIterationsParameter {
get { return (IFixedValueParameter)Parameters["LocalImprovementMaximumIterations"]; }
}
private ConstrainedValueParameter LocalImprovementParameter {
get { return (ConstrainedValueParameter)Parameters["LocalImprovement"]; }
}
public IFixedValueParameter MinimumEliteSetSizeParameter {
get { return (IFixedValueParameter)Parameters["MinimumEliteSetSize"]; }
}
public ConstrainedValueParameter PathRelinkingParameter {
get { return (ConstrainedValueParameter)Parameters["PathRelinking"]; }
}
public ConstrainedValueParameter EliteSetReducerParameter {
get { return (ConstrainedValueParameter)Parameters["EliteSetReducer"]; }
}
#endregion
#region Properties
public BoolValue SetSeedRandomly {
get { return SetSeedRandomlyParameter.Value; }
set { SetSeedRandomlyParameter.Value = value; }
}
public IntValue Seed {
get { return SeedParameter.Value; }
set { SeedParameter.Value = value; }
}
public MultiAnalyzer Analyzer {
get { return AnalyzerParameter.Value; }
set { AnalyzerParameter.Value = value; }
}
public IntValue EliteSetSize {
get { return EliteSetSizeParameter.Value; }
set { EliteSetSizeParameter.Value.Value = value.Value; }
}
public IntValue LocalImprovementMaximumIterations {
get { return LocalImprovementMaximumIterationsParameter.Value; }
set { LocalImprovementMaximumIterationsParameter.Value.Value = value.Value; }
}
private RandomCreator RandomCreator {
get { return (RandomCreator)OperatorGraph.InitialOperator; }
}
private VariableCreator VariableCreator {
get { return (VariableCreator)RandomCreator.Successor; }
}
private SolutionsCreator SolutionsCreator {
get { return (SolutionsCreator)VariableCreator.Successor; }
}
private SubScopesCounter SubScopesCounter {
get { return (SubScopesCounter)SolutionsCreator.Successor; }
}
private ResultsCollector ResultsCollector {
get { return (ResultsCollector)SubScopesCounter.Successor; }
}
private GRASPWithPathRelinkingMainLoop MainLoop {
get { return (GRASPWithPathRelinkingMainLoop)ResultsCollector.Successor; }
}
#endregion
[Storable]
private BestAverageWorstQualityAnalyzer analyzer;
[StorableConstructor]
private GRASPWithPathRelinking(bool deserializing) : base(deserializing) { }
private GRASPWithPathRelinking(GRASPWithPathRelinking original, Cloner cloner)
: base(original, cloner) {
analyzer = cloner.Clone(original.analyzer);
RegisterEventHandlers();
}
public GRASPWithPathRelinking()
: base() {
Parameters.Add(new ValueParameter("SetSeedRandomly", "True if the random seed should be set to a random value, otherwise false.", new BoolValue(true)));
Parameters.Add(new ValueParameter("Seed", "The random seed used to initialize the new pseudo random number generator."));
Parameters.Add(new ValueParameter("Analyzer", "The operator used to analyze each iteration.", new MultiAnalyzer()));
Parameters.Add(new ValueParameter("MaximumIterations", "The maximum number of iterations that the algorithm should run.", new IntValue(1000)));
Parameters.Add(new FixedValueParameter("EliteSetSize", "The elite set stores the best found solutions.", new IntValue(10)));
Parameters.Add(new FixedValueParameter("LocalImprovementMaximumIterations", "The maximum number of iterations performed by the local improvement operator.", new IntValue(1000)));
Parameters.Add(new ConstrainedValueParameter("LocalImprovement", "Performs a local search on the solution."));
Parameters.Add(new FixedValueParameter("MinimumEliteSetSize", "(ρ) The minimum amount of elites for performing path relinking.", new IntValue(2)));
Parameters.Add(new ConstrainedValueParameter("PathRelinking", "The operator that performs the path relinking."));
Parameters.Add(new ConstrainedValueParameter("EliteSetReducer", "The operator that reduces the old elite set and the new solution(s) to a new elite set."));
analyzer = new BestAverageWorstQualityAnalyzer();
Analyzer.Operators.Add(analyzer);
var randomCreator = new RandomCreator();
OperatorGraph.InitialOperator = randomCreator;
randomCreator.RandomParameter.ActualName = "Random";
randomCreator.SeedParameter.ActualName = SeedParameter.Name;
randomCreator.SeedParameter.Value = null;
randomCreator.SetSeedRandomlyParameter.ActualName = SetSeedRandomlyParameter.Name;
randomCreator.SetSeedRandomlyParameter.Value = null;
var variableCreator = new VariableCreator();
variableCreator.CollectedValues.Add(new ValueParameter("EvaluatedSolutions", new IntValue(0)));
variableCreator.CollectedValues.Add(new ValueParameter("Iterations", new IntValue(0)));
randomCreator.Successor = variableCreator;
var solutionsCreator = new SolutionsCreator();
solutionsCreator.NumberOfSolutionsParameter.ActualName = MinimumEliteSetSizeParameter.Name;
solutionsCreator.ParallelParameter.Value = new BoolValue(true);
variableCreator.Successor = solutionsCreator;
var subscopesCounter = new SubScopesCounter();
subscopesCounter.Name = "EvaluatedSolutions++";
subscopesCounter.AccumulateParameter.Value.Value = true;
subscopesCounter.ValueParameter.ActualName = "EvaluatedSolutions";
solutionsCreator.Successor = subscopesCounter;
var resultsCollector = new ResultsCollector();
resultsCollector.CopyValue.Value = false;
resultsCollector.CollectedValues.Add(new LookupParameter("EvaluatedSolutions", "Counter for the number of times the evaluation function is called."));
resultsCollector.CollectedValues.Add(new LookupParameter("Iterations", "The algorithm's current iteration"));
subscopesCounter.Successor = resultsCollector;
var mainLoop = new GRASPWithPathRelinkingMainLoop();
mainLoop.AnalyzerParameter.ActualName = AnalyzerParameter.Name;
mainLoop.EliteSetReducerParameter.ActualName = EliteSetReducerParameter.Name;
mainLoop.EliteSetSizeParameter.ActualName = EliteSetSizeParameter.Name;
mainLoop.EvaluatedSolutionsParameter.ActualName = "EvaluatedSolutions";
mainLoop.LocalImprovementParameter.ActualName = LocalImprovementParameter.Name;
mainLoop.MaximumIterationsParameter.ActualName = MaximumIterationsParameter.Name;
mainLoop.PathRelinkingParameter.ActualName = PathRelinkingParameter.Name;
mainLoop.ResultsParameter.ActualName = "Results";
resultsCollector.Successor = mainLoop;
InitializeOperators();
RegisterEventHandlers();
}
public override IDeepCloneable Clone(Cloner cloner) {
return new GRASPWithPathRelinking(this, cloner);
}
public override void Prepare() {
if (Problem != null) base.Prepare();
}
#region Events
protected override void OnProblemChanged() {
InitializeOperators();
base.OnProblemChanged();
}
protected override void Problem_SolutionCreatorChanged(object sender, EventArgs e) {
Parameterize();
base.Problem_SolutionCreatorChanged(sender, e);
}
protected override void Problem_EvaluatorChanged(object sender, EventArgs e) {
Parameterize();
base.Problem_EvaluatorChanged(sender, e);
}
protected override void Problem_OperatorsChanged(object sender, EventArgs e) {
InitializeOperators();
base.Problem_OperatorsChanged(sender, e);
}
private void LocalImprovementParameter_ValueChanged(object sender, EventArgs e) {
Parameterize();
}
#endregion
#region Helpers
[StorableHook(HookType.AfterDeserialization)]
private void AfterDeserialization() {
if (Problem != null) {
}
RegisterEventHandlers();
}
private void RegisterEventHandlers() {
LocalImprovementParameter.ValueChanged += new EventHandler(LocalImprovementParameter_ValueChanged);
}
private void InitializeOperators() {
Analyzer.Operators.Clear();
if (Problem != null) {
foreach (IAnalyzer a in Problem.Operators.OfType()) {
foreach (var param in a.Parameters.OfType())
param.Depth = 1;
Analyzer.Operators.Add(a);
Analyzer.Operators.SetItemCheckedState(a, a.EnabledByDefault);
}
InitializeFromInstallation(LocalImprovementParameter, x => x is ILocalImprovementAlgorithmOperator
&& ((ILocalImprovementAlgorithmOperator)x).ProblemType.IsInstanceOfType(Problem));
InitializeFromProblem(PathRelinkingParameter);
InitializeFromProblem(EliteSetReducerParameter);
} else {
LocalImprovementParameter.ValidValues.Clear();
PathRelinkingParameter.ValidValues.Clear();
EliteSetReducerParameter.ValidValues.Clear();
}
Analyzer.Operators.Add(analyzer);
Parameterize();
}
private void Parameterize() {
if (Problem != null) {
SolutionsCreator.EvaluatorParameter.ActualName = Problem.EvaluatorParameter.Name;
SolutionsCreator.SolutionCreatorParameter.ActualName = Problem.SolutionCreatorParameter.Name;
MainLoop.EvaluatorParameter.ActualName = Problem.EvaluatorParameter.Name;
MainLoop.SolutionCreatorParameter.ActualName = Problem.SolutionCreatorParameter.Name;
analyzer.QualityParameter.ActualName = Problem.Evaluator.QualityParameter.ActualName;
if (LocalImprovementParameter.Value != null) {
if (LocalImprovementParameter.Value is ILocalImprovementAlgorithmOperator)
((ILocalImprovementAlgorithmOperator)LocalImprovementParameter.Value).Problem = Problem;
}
}
foreach (var localImprovement in LocalImprovementParameter.ValidValues) {
localImprovement.MaximumIterationsParameter.ActualName = LocalImprovementMaximumIterationsParameter.Name;
localImprovement.EvaluatedSolutionsParameter.ActualName = "EvaluatedSolutions";
localImprovement.ResultsParameter.ActualName = "Results";
}
foreach (var reducer in EliteSetReducerParameter.ValidValues) {
reducer.MinimumPopulationSizeParameter.ActualName = MinimumEliteSetSizeParameter.Name;
reducer.MaximumPopulationSizeParameter.ActualName = EliteSetSizeParameter.Name;
}
}
private void InitializeFromProblem(ConstrainedValueParameter parameter) where T : class, INamedItem {
InitializeFromProblem(parameter, x => true);
}
private void InitializeFromProblem(ConstrainedValueParameter parameter, Func condition) where T : class, INamedItem {
if (parameter == null) throw new ArgumentNullException("parameter");
if (condition == null) throw new ArgumentNullException("condition");
string oldName = String.Empty;
if (parameter.Value != null) oldName = parameter.Value.Name;
parameter.ValidValues.Clear();
foreach (var item in Problem.Operators.OfType().Where(condition)) {
parameter.ValidValues.Add(item);
}
var newItem = parameter.ValidValues.FirstOrDefault(x => x.Name == oldName);
if (newItem != null) parameter.Value = newItem;
}
private void InitializeFromInstallation(ConstrainedValueParameter parameter) where T : class, INamedItem {
InitializeFromInstallation(parameter, x => true);
}
private void InitializeFromInstallation(ConstrainedValueParameter parameter, Func condition) where T : class, INamedItem {
if (parameter == null) throw new ArgumentNullException("parameter");
if (condition == null) throw new ArgumentNullException("condition");
string oldName = String.Empty;
if (parameter.Value != null) oldName = parameter.Value.Name;
parameter.ValidValues.Clear();
foreach (var item in ApplicationManager.Manager.GetInstances().Where(condition)) {
parameter.ValidValues.Add(item);
}
var newItem = parameter.ValidValues.FirstOrDefault(x => x.Name == oldName);
if (newItem != null) parameter.Value = newItem;
}
#endregion
}
}