#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 System; using System.Collections.Generic; using System.Linq; using HeuristicLab.Common; using HeuristicLab.Core; using HeuristicLab.Data; using HeuristicLab.Optimization; using HeuristicLab.Optimization.Operators; using HeuristicLab.Parameters; using HeuristicLab.Persistence.Default.CompositeSerializers.Storable; using HeuristicLab.Problems.VehicleRouting.Encodings.Potvin; using HeuristicLab.Problems.VehicleRouting.Interfaces; using HeuristicLab.Problems.VehicleRouting.ProblemInstances; using HeuristicLab.Problems.VehicleRouting.Variants; namespace HeuristicLab.Problems.VehicleRouting { /// /// An operator that relinks paths between vehicle routing solutions. /// [Item("VRPPathRelinker", "An operator that relinks paths between vehicle routing solutions.")] [StorableClass] public sealed class VRPPathRelinker : SingleObjectivePathRelinker, IGeneralVRPOperator, IStochasticOperator { #region Parameter properties public ILookupParameter RandomParameter { get { return (ILookupParameter)Parameters["Random"]; } } public ILookupParameter ProblemInstanceParameter { get { return (ILookupParameter)Parameters["ProblemInstance"]; } } public IValueParameter SampleSizeParameter { get { return (IValueParameter)Parameters["SampleSize"]; } } public IValueParameter IterationsParameter { get { return (IValueParameter)Parameters["Iterations"]; } } #endregion [StorableConstructor] private VRPPathRelinker(bool deserializing) : base(deserializing) { } private VRPPathRelinker(VRPPathRelinker original, Cloner cloner) : base(original, cloner) { } public VRPPathRelinker() : base() { Parameters.Add(new LookupParameter("Random", "The pseudo random number generator which should be used for stochastic manipulation operators.")); Parameters.Add(new LookupParameter("ProblemInstance", "The VRP problem instance")); Parameters.Add(new ValueParameter("SampleSize", "The number of moves that should be executed.", new IntValue(10))); Parameters.Add(new ValueParameter("Iterations", "The number of iterations the operator should perform.", new IntValue(50))); } public override IDeepCloneable Clone(Cloner cloner) { return new VRPPathRelinker(this, cloner); } private static int MatchingCities(Tour tour1, Tour tour2) { return tour1.Stops.Intersect(tour2.Stops).Count(); } private static PotvinEncoding MatchTours(PotvinEncoding initiator, PotvinEncoding guide, IVRPProblemInstance problemInstance) { PotvinEncoding result = new PotvinEncoding(problemInstance); List used = new List(); for (int i = 0; i < initiator.Tours.Count; i++) { used.Add(false); } for (int i = 0; i < guide.Tours.Count; i++) { int bestMatch = -1; int bestTour = -1; for (int j = 0; j < initiator.Tours.Count; j++) { if (!used[j]) { int match = MatchingCities(guide.Tours[i], initiator.Tours[j]); if (match > bestMatch) { bestMatch = match; bestTour = j; } } } if (bestTour != -1) { result.Tours.Add(initiator.Tours[bestTour]); used[bestTour] = true; } } for (int i = 0; i < initiator.Tours.Count; i++) { if (!used[i]) result.Tours.Add(initiator.Tours[i]); } return result; } #region moves public static PotvinEncoding RouteBasedXOver(PotvinEncoding initiator, PotvinEncoding guide, IRandom random, IVRPProblemInstance problemInstance) { return PotvinRouteBasedCrossover.Apply(random, initiator, guide, problemInstance, false); } public static PotvinEncoding SequenceBasedXOver(PotvinEncoding initiator, PotvinEncoding guide, IRandom random, IVRPProblemInstance problemInstance) { return PotvinSequenceBasedCrossover.Apply(random, initiator, guide, problemInstance, false); } public static void GuidedRelocateMove(PotvinEncoding initiator, PotvinEncoding guide, IRandom random) { List cities = new List(); foreach (Tour tour in initiator.Tours) { foreach (int city in tour.Stops) { Tour guideTour = guide.Tours.First(t => t.Stops.Contains(city)); if (guide.Tours.IndexOf(guideTour) != initiator.Tours.IndexOf(tour)) { cities.Add(city); } } } if (cities.Count == 0) { RelocateMove(initiator, random); } else { int city = cities[random.Next(cities.Count)]; Tour tour = initiator.Tours.First(t => t.Stops.Contains(city)); tour.Stops.Remove(city); Tour guideTour = guide.Tours.First(t => t.Stops.Contains(city)); int guideTourIndex = guide.Tours.IndexOf(guideTour); if (guideTourIndex < initiator.Tours.Count) { Tour tour2 = initiator.Tours[guideTourIndex]; int guideIndex = guideTour.Stops.IndexOf(city); if (guideIndex == 0) { tour2.Stops.Insert(0, city); } else { int predecessor = guideTour.Stops[guideIndex - 1]; int initIndex = tour2.Stops.IndexOf(predecessor); if (initIndex != -1) { tour2.Stops.Insert(initIndex + 1, city); } else { if (guideIndex == guideTour.Stops.Count - 1) { tour2.Stops.Insert(tour2.Stops.Count, city); } else { int sucessor = guideTour.Stops[guideIndex + 1]; initIndex = tour2.Stops.IndexOf(sucessor); if (initIndex != -1) { tour2.Stops.Insert(initIndex, city); } else { tour2.Stops.Insert(random.Next(tour2.Stops.Count + 1), city); } } } } } else { Tour tour2 = new Tour(); tour2.Stops.Add(city); initiator.Tours.Add(tour2); } if (tour.Stops.Count == 0) initiator.Tours.Remove(tour); } } public static void RelocateMove(PotvinEncoding individual, IRandom random) { int cities = individual.Cities; int city = 1 + random.Next(cities); Tour originalTour = individual.Tours.Find(t => t.Stops.Contains(city)); //consider creating new route individual.Tours.Add(new Tour()); int position = 1 + random.Next(cities + individual.Tours.Count - 1); if (position >= city) { position++; } int originalPosition = originalTour.Stops.IndexOf(city); originalTour.Stops.RemoveAt(originalPosition); Tour insertionTour; int insertionPosition; if (position <= cities) { insertionTour = individual.Tours.Find(t => t.Stops.Contains(position)); insertionPosition = insertionTour.Stops.IndexOf(position) + 1; } else { insertionTour = individual.Tours[position - cities - 1]; insertionPosition = 0; } insertionTour.Stops.Insert(insertionPosition, city); individual.Tours.RemoveAll(t => t.Stops.Count == 0); } public static void ExchangeMove(PotvinEncoding individual, IRandom random) { if (individual.Tours.Count > 1) { int tour1Idx = random.Next(individual.Tours.Count); int tour2Idx = random.Next(individual.Tours.Count - 1); if (tour2Idx >= tour1Idx) tour2Idx++; Tour tour1 = individual.Tours[tour1Idx]; Tour tour2 = individual.Tours[tour2Idx]; int index1 = random.Next(tour1.Stops.Count); int city1 = tour1.Stops[index1]; int index2 = random.Next(tour2.Stops.Count); int city2 = tour2.Stops[index2]; tour1.Stops.RemoveAt(index1); tour1.Stops.Insert(index1, city2); tour2.Stops.RemoveAt(index2); tour2.Stops.Insert(index2, city1); } } public static void TwoOptMove(PotvinEncoding individual, IRandom random) { List tours = individual.Tours.FindAll(t => t.Stops.Count >= 4); if (tours.Count > 0) { int tourIdx = random.Next(tours.Count); Tour tour = tours[tourIdx]; int a; if (tour.Stops.Count == 4) { a = 0; } else if (tour.Stops.Count == 5) { int idx = random.Next(4); if (idx >= 2) idx++; a = idx; } else { a = random.Next(tour.Stops.Count); } int b; List indices = new List(); for (int i = 0; i < tour.Stops.Count; i++) { if (Math.Abs(i - a) > 2) { indices.Add(i); } } b = indices[random.Next(indices.Count)]; if (b < a) { int tmp = a; a = b; b = tmp; } int index = a + 1; int count = b - a - 1; List segment = tour.Stops.GetRange(index, count); tour.Stops.RemoveRange(index, count); segment.Reverse(); tour.Stops.InsertRange(index, segment); } } public static void TwoOptStarMove(PotvinEncoding individual, IRandom random) { //consider creating new tour individual.Tours.Add(new Tour()); int route1Idx = random.Next(individual.Tours.Count); int route2Idx = random.Next(individual.Tours.Count - 1); if (route2Idx >= route1Idx) route2Idx++; Tour route1 = individual.Tours[route1Idx]; Tour route2 = individual.Tours[route2Idx]; int x1 = random.Next(route1.Stops.Count + 1); int x2 = random.Next(route2.Stops.Count + 1); int count = route1.Stops.Count - x1; List segmentX1 = new List(); if (count > 0) { segmentX1 = route1.Stops.GetRange(x1, count); route1.Stops.RemoveRange(x1, count); } count = route2.Stops.Count - x2; List segmentX2 = new List(); if (count > 0) { segmentX2 = route2.Stops.GetRange(x2, count); route2.Stops.RemoveRange(x2, count); } route1.Stops.AddRange(segmentX2); route2.Stops.AddRange(segmentX1); individual.Tours.RemoveAll(t => t.Stops.Count == 0); } public static void OrOptMove(PotvinEncoding individual, IRandom random) { List tours = individual.Tours.FindAll(t => t.Stops.Count >= 2); if (tours.Count > 0) { int tourIdx = random.Next(tours.Count); Tour tour = tours[tourIdx]; int segmentStart = random.Next(tour.Stops.Count); int segmentLength; if (segmentStart == 0) { segmentLength = 1 + random.Next(tour.Stops.Count - 1); } else { segmentLength = 1 + random.Next(tour.Stops.Count - segmentStart); } List segment = tour.Stops.GetRange(segmentStart, segmentLength); tour.Stops.RemoveRange(segmentStart, segmentLength); int newPos; if (tour.Stops.Count == 1) newPos = 0; else newPos = random.Next(tour.Stops.Count - 1); if (newPos >= segmentStart) newPos++; tour.Stops.InsertRange(newPos, segment); } } #endregion private static IList ChooseSelection(IList solutions, PercentValue n) { IList selection = new List(); if (solutions.Count > 0) { int noSol = (int)(solutions.Count * n.Value); if (noSol <= 0) noSol++; double stepSize = (double)solutions.Count / (double)noSol; for (int i = 0; i < noSol; i++) selection.Add(solutions.ElementAt((int)((i + 1) * stepSize - stepSize * 0.5))); } return selection; } public static ItemArray Apply(PotvinEncoding initiator, PotvinEncoding guide, PercentValue n, int sampleSize, int iterations, IRandom rand, IVRPProblemInstance problemInstance) { if (initiator == null || guide == null) throw new ArgumentException("Cannot relink path because one of the provided solutions or both are null."); double sigma = 1.5; DoubleValue originalOverloadPenalty = new DoubleValue(); if (problemInstance is IHomogenousCapacitatedProblemInstance) originalOverloadPenalty.Value = (problemInstance as IHomogenousCapacitatedProblemInstance).OverloadPenalty.Value; DoubleValue originalTardinessPenalty = new DoubleValue(); if (problemInstance is ITimeWindowedProblemInstance) originalTardinessPenalty.Value = (problemInstance as ITimeWindowedProblemInstance).TardinessPenalty.Value; PotvinEncoding current = MatchTours(initiator, guide, problemInstance); double currentSimilarity = VRPSimilarityCalculator.CalculateSimilarity(current, guide); IList solutions = new List(); int i = 0; while (i < iterations && currentSimilarity != 1.0) { VRPEvaluation currentEval = problemInstance.Evaluate(current); currentSimilarity = VRPSimilarityCalculator.CalculateSimilarity(current, guide); if (currentSimilarity < 1.0) { for (int sample = 0; sample < sampleSize; sample++) { PotvinEncoding next = current.Clone() as PotvinEncoding; int neighborhood = rand.Next(4); switch (neighborhood) { case 0: next = RouteBasedXOver(next, guide, rand, problemInstance); break; case 1: next = SequenceBasedXOver(next, guide, rand, problemInstance); break; case 2: TwoOptMove(next, rand); break; case 3: GuidedRelocateMove(next, guide, rand); break; } next = MatchTours(next, guide, problemInstance); VRPEvaluation nextEval = problemInstance.Evaluate(next); double nextSimilarity = VRPSimilarityCalculator.CalculateSimilarity(next, guide); if (nextSimilarity > currentSimilarity && nextEval.Quality <= currentEval.Quality) { current = next; solutions.Add(current); break; } } if (problemInstance is IHomogenousCapacitatedProblemInstance) { if (((CVRPEvaluation)currentEval).Overload > 0) (problemInstance as IHomogenousCapacitatedProblemInstance).OverloadPenalty.Value *= sigma; else (problemInstance as IHomogenousCapacitatedProblemInstance).OverloadPenalty.Value /= sigma; } if (problemInstance is ITimeWindowedProblemInstance) { if (((CVRPTWEvaluation)currentEval).Tardiness > 0) (problemInstance as ITimeWindowedProblemInstance).TardinessPenalty.Value *= sigma; else (problemInstance as ITimeWindowedProblemInstance).TardinessPenalty.Value /= sigma; } i++; } } if (problemInstance is IHomogenousCapacitatedProblemInstance) (problemInstance as IHomogenousCapacitatedProblemInstance).OverloadPenalty.Value = originalOverloadPenalty.Value; if (problemInstance is ITimeWindowedProblemInstance) (problemInstance as ITimeWindowedProblemInstance).TardinessPenalty.Value = originalTardinessPenalty.Value; return new ItemArray(ChooseSelection(solutions, n)); } protected override ItemArray Relink(ItemArray parents, PercentValue n) { if (parents.Length != 2) throw new ArgumentException("The number of parents is not equal to 2."); if (!(parents[0] is PotvinEncoding)) parents[0] = PotvinEncoding.ConvertFrom(parents[0] as IVRPEncoding, ProblemInstanceParameter.ActualValue); if (!(parents[1] is PotvinEncoding)) parents[1] = PotvinEncoding.ConvertFrom(parents[1] as IVRPEncoding, ProblemInstanceParameter.ActualValue); return Apply(parents[0] as PotvinEncoding, parents[1] as PotvinEncoding, n, SampleSizeParameter.Value.Value, IterationsParameter.Value.Value, RandomParameter.ActualValue, ProblemInstanceParameter.ActualValue); } } }