#region License Information
/* HeuristicLab
* Copyright (C) 2002-2014 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.Drawing;
using System.Linq;
using System.Threading;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Core.Networks;
using HeuristicLab.Data;
using HeuristicLab.Encodings.BinaryVectorEncoding;
using HeuristicLab.Encodings.PermutationEncoding;
using HeuristicLab.Networks.Programmable;
using HeuristicLab.Problems.Knapsack;
using HeuristicLab.Problems.TravelingSalesman;
namespace HeuristicLab.Networks {
[Item("KSPTSPControl", "A node of an optimization network which connects a KSP and a TSP.")]
public class CompiledKSPTSPControl : ProgrammableNode.CompiledProgrammableNode {
public static new Image StaticItemImage {
get { return HeuristicLab.Common.Resources.VSImageLibrary.RadialChart; }
}
new protected KSPTSPControl Context {
get { return (KSPTSPControl)base.Context; }
}
protected CompiledKSPTSPControl(CompiledKSPTSPControl original, Cloner cloner) : base(original, cloner) { }
public CompiledKSPTSPControl(ProgrammableNode context)
: base(context) {
if (Ports.Count == 0)
Initialize();
}
public override IDeepCloneable Clone(Cloner cloner) {
return new CompiledKSPTSPControl(this, cloner);
}
public override void Initialize() {
base.Initialize();
var configPort = new ConfigurationPort("Configure");
Ports.Add(configPort);
configPort.Parameters.Add(new PortParameter("KnapsackCapacity") {
Type = PortParameterType.Input
});
configPort.Parameters.Add(new PortParameter("Values") {
Type = PortParameterType.Input
});
configPort.Parameters.Add(new PortParameter("Weights") {
Type = PortParameterType.Input
});
configPort.Parameters.Add(new PortParameter("Coordinates") {
Type = PortParameterType.Input
});
configPort.Parameters.Add(new PortParameter("TransportCostsFactor") {
Type = PortParameterType.Input
});
var evalKspPort = new MessagePort("Evaluate KSP");
Ports.Add(evalKspPort);
evalKspPort.Parameters.Add(new PortParameter("KnapsackSolution") {
Type = PortParameterType.Input
});
evalKspPort.Parameters.Add(new PortParameter("Quality") {
Type = PortParameterType.Output
});
var evalTspPort = new MessagePort("Evaluate TSP");
Ports.Add(evalTspPort);
evalTspPort.Parameters.Add(new PortParameter("TSPTour") {
Type = PortParameterType.Input
});
evalTspPort.Parameters.Add(new PortParameter("TSPTourLength") {
Type = PortParameterType.Output
});
var addKspSolultionPort = new MessagePort("Add KSP Solution");
Ports.Add(addKspSolultionPort);
addKspSolultionPort.Parameters.Add(new PortParameter("BestSolution") {
Type = PortParameterType.Input
});
var addTspSolutionPort = new MessagePort("Add TSP Solution");
Ports.Add(addTspSolutionPort);
addTspSolutionPort.Parameters.Add(new PortParameter("BestSolution") {
Type = PortParameterType.Input
});
}
private void AddKspSolultionPortOnMessageReceived(object sender, EventArgs e) {
var cities = (KnapsackSolution)(e.Value.Values["BestSolution"]).Value;
AddSelectedCities(cities.BinaryVector);
}
private void AddTspSolutionPortOnMessageReceived(object sender, EventArgs e) {
var trip = (PathTSPTour)(e.Value.Values["BestSolution"]).Value;
AddPredefinedTrip(trip.Permutation);
}
private void ConfigPortOnMessageReceived(object sender, EventArgs e) {
Context.TransportCostFactor = (DoubleValue)(e.Value["TransportCostsFactor"]);
Context.Coordinates = (DoubleMatrix)(e.Value["Coordinates"]);
Context.Distances = CalculateEuclidean(Context.Coordinates);
Context.CityValues = (IntArray)(e.Value["Values"]);
Context.CityWeights = (IntArray)(e.Value["Weights"]);
Context.CityLimit = (IntValue)(e.Value["KnapsackCapacity"]);
}
private void EvalKspPortOnMessageReceived(object sender, EventArgs e) {
var cities = (BinaryVector)(e.Value.Values["KnapsackSolution"]).Value;
e.Value.Values["Quality"].Value = new DoubleValue(EvaluatePredefinedTrip(cities));
}
private void EvalTspPortOnMessageReceived(object sender, EventArgs e) {
var trip = (Permutation)(e.Value.Values["TSPTour"]).Value;
e.Value.Values["TSPTourLength"].Value = new DoubleValue(EvaluateSelectedCities(trip));
}
public override void RegisterEvents() {
base.RegisterEvents();
((IMessagePort)Ports["Configure"]).MessageReceived += ConfigPortOnMessageReceived;
((IMessagePort)Ports["Evaluate KSP"]).MessageReceived += EvalKspPortOnMessageReceived;
((IMessagePort)Ports["Evaluate TSP"]).MessageReceived += EvalTspPortOnMessageReceived;
((IMessagePort)Ports["Add KSP Solution"]).MessageReceived += AddKspSolultionPortOnMessageReceived;
((IMessagePort)Ports["Add TSP Solution"]).MessageReceived += AddTspSolutionPortOnMessageReceived;
}
public override void DeregisterEvents() {
((IMessagePort)Ports["Configure"]).MessageReceived -= ConfigPortOnMessageReceived;
((IMessagePort)Ports["Evaluate KSP"]).MessageReceived -= EvalKspPortOnMessageReceived;
((IMessagePort)Ports["Evaluate TSP"]).MessageReceived -= EvalTspPortOnMessageReceived;
((IMessagePort)Ports["Add KSP Solution"]).MessageReceived -= AddKspSolultionPortOnMessageReceived;
((IMessagePort)Ports["Add TSP Solution"]).MessageReceived -= AddTspSolutionPortOnMessageReceived;
base.DeregisterEvents();
}
public double EvaluatePredefinedTrip(BinaryVector cities) {
if (Context.SelectedCities.Count == 0) {
Context.SelectedCities.Add(cities);
Context.KspWait.Set();
Context.TspWait.WaitOne();
}
return EvaluateBoth(cities, Context.PredefinedTrip.Last());
}
public double EvaluateSelectedCities(Permutation trip) {
if (Context.PredefinedTrip.Count == 0) {
Context.PredefinedTrip.Add(trip);
Context.TspWait.Set();
Context.KspWait.WaitOne();
}
return EvaluateBoth(Context.SelectedCities.Last(), trip);
}
public double EvaluateBoth(BinaryVector cities, Permutation trip) {
var cityValues = cities.Select((v, i) => v ? Context.CityValues[i] : 0).Sum();
var cityWeights = cities.Select((v, i) => v ? Context.CityWeights[i] : 0).Sum();
var subtour = trip.Where(x => cities[x]).ToArray();
var tourLength = 0.0;
for (var i = 1; i < subtour.Length; i++)
tourLength += Context.Distances[subtour[i - 1], subtour[i]];
tourLength += Context.Distances[subtour.Last(), subtour[0]];
if (cityWeights > Context.CityLimit.Value) // infeasible solution
return Context.CityLimit.Value - cityWeights - tourLength * Context.TransportCostFactor.Value;
return cityValues - tourLength * Context.TransportCostFactor.Value;
}
public void AddPredefinedTrip(Permutation trip) {
Context.TspWait.Set();
Context.KspWait.WaitOne();
lock (Context.Locker) {
Context.PredefinedTrip.Add(trip);
}
}
public void AddSelectedCities(BinaryVector cities) {
Context.KspWait.Set();
Context.TspWait.WaitOne();
lock (Context.Locker) {
Context.SelectedCities.Add(cities);
}
}
public static DoubleMatrix CalculateEuclidean(DoubleMatrix cities) {
var len = cities.Rows;
var distances = new DoubleMatrix(len, len);
for (var i = 0; i < len - 1; i++) {
var sX = cities[i, 0];
var sY = cities[i, 1];
for (var j = i + 1; j < len; j++) {
var tX = cities[j, 0];
var tY = cities[j, 1];
distances[i, j] = Math.Sqrt((sX - tX) * (sX - tX) + (sY - tY) * (sY - tY));
distances[j, i] = distances[i, j];
}
}
return distances;
}
}
}