#region License Information /* HeuristicLab * Copyright (C) 2002-2016 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.Linq; using HeuristicLab.Common; using HeuristicLab.Core; using HeuristicLab.Data; using HeuristicLab.Optimization; using HeuristicLab.Parameters; using HeuristicLab.Persistence.Default.CompositeSerializers.Storable; using HeuristicLab.Problems.Instances; namespace HeuristicLab.Problems.BinPacking2D { [StorableClass] public abstract class ProblemBase : SingleObjectiveBasicProblem, IProblemInstanceConsumer, IProblemInstanceExporter where TEnc : class, IEncoding where TSol : class, IItem { protected readonly string SolutionEvaluatorParameterName = "SolutionEvaluator"; public readonly string EncodedSolutionName = "EncodedSolution"; #region Default Instance private readonly BPPData defaultInstance = new BPPData() { Name = "2D BPP Default Instance", Description = "The default instance for 2D Bin Packing.", BinShape = new PackingShape(20, 16), Items = new PackingItem[] { new PackingItem(3, 8, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(5, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(9, 3, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0}, new PackingItem(2, 7, new PackingShape(20, 16)) { Layer = 1, Weight = 1.0} }, }; #endregion #region parameter properties public IValueParameter> DecoderParameter { get { return (IValueParameter>)Parameters["Decoder"]; } } public IValueParameter SolutionEvaluatorParameter { get { return (IValueParameter)Parameters[SolutionEvaluatorParameterName]; } } public IValueParameter> ItemsParameter { get { return (IValueParameter>)Parameters["Items"]; } } public IValueParameter BinShapeParameter { get { return (IValueParameter)Parameters["BinShape"]; } } public IValueParameter BestKnownSolutionParameter { get { return (IValueParameter)Parameters["BestKnownSolution"]; } } public IFixedValueParameter LowerBoundParameter { get { return (IFixedValueParameter)Parameters["LowerBound"]; } } #endregion #region properties public IDecoder Decoder { get { return DecoderParameter.Value; } set { DecoderParameter.Value = value; } } public IEvaluator SolutionEvaluator { get { return SolutionEvaluatorParameter.Value; } set { SolutionEvaluatorParameter.Value = value; } } public ReadOnlyItemList Items { get { return ItemsParameter.Value; } set { ItemsParameter.Value = value; } } public PackingShape BinShape { get { return BinShapeParameter.Value; } set { BinShapeParameter.Value = value; } } public Solution BestKnownSolution { get { return BestKnownSolutionParameter.Value; } set { BestKnownSolutionParameter.Value = value; } } public int LowerBound { get { return LowerBoundParameter.Value.Value; } } public int NumberOfItems { get { return Items == null ? 0 : Items.Count; } } #endregion // persistence [StorableConstructor] protected ProblemBase(bool deserializing) : base(deserializing) { } // cloning protected ProblemBase(ProblemBase original, Cloner cloner) : base(original, cloner) { } protected ProblemBase() : base() { var defaultEvaluator = new PackingRatioEvaluator(); Parameters.Add(new ValueParameter>("Decoder", "The decoder translates a permutation to a packing solution candidiates")); Parameters.Add(new ValueParameter(SolutionEvaluatorParameterName, "The evaluator calculates qualities of solution candidates", defaultEvaluator)); Parameters.Add(new ValueParameter>("Items", "The items which must be packed into bins")); Parameters.Add(new ValueParameter("BinShape", "The size of bins into which items must be packed")); Parameters.Add(new OptionalValueParameter("BestKnownSolution", "The best solution found so far")); Parameters.Add(new FixedValueParameter("LowerBound", "A lower bound for the number of bins that is necessary to pack all items")); Load(defaultInstance); } public override bool Maximization { get { return true; } } public override double Evaluate(Individual individual, IRandom random) { var encodedSolutionCand = (TSol)individual[EncodedSolutionName]; var decoder = Decoder; var solution = decoder.Decode(encodedSolutionCand, BinShape, Items); return SolutionEvaluator.Evaluate(solution); } public override void Analyze(Individual[] individuals, double[] qualities, ResultCollection results, IRandom random) { base.Analyze(individuals, qualities, results, random); Analyze(individuals.Select(i => (TSol)i[EncodedSolutionName]).ToArray(), qualities, results, random); } public virtual void Analyze(TSol[] individuals, double[] qualities, ResultCollection results, IRandom random) { var bestSolutionResultName = "Best Packing Solution"; var numContainersResultName = "Nr of Containers"; var binUtilResultName = "Overall Bin Utilization"; if (!results.ContainsKey(bestSolutionResultName)) results.Add(new Result(bestSolutionResultName, typeof(Solution))); if (!results.ContainsKey(numContainersResultName)) results.Add(new Result(numContainersResultName, typeof(IntValue))); if (!results.ContainsKey(binUtilResultName)) results.Add(new Result(binUtilResultName, typeof(DoubleValue))); // find index of item with max quality int bestIdx = 0; for (int j = 1; j < qualities.Length; j++) if (qualities[j] > qualities[bestIdx]) bestIdx = j; // update best solution so far var bestSolution = results[bestSolutionResultName].Value as Solution; if (bestSolution == null || bestSolution.Quality.Value < qualities[bestIdx]) { var newBestSolution = Decoder.Decode(individuals[bestIdx], BinShape, Items); newBestSolution.Quality = new DoubleValue(qualities[bestIdx]); results[bestSolutionResultName].Value = newBestSolution; results[numContainersResultName].Value = new IntValue(newBestSolution.NrOfBins); results[binUtilResultName].Value = new DoubleValue(BinUtilizationEvaluator.CalculateBinUtilization(newBestSolution)); // update best known solution var bestKnownQuality = BestKnownQualityParameter.Value; if (bestKnownQuality == null || bestKnownQuality.Value < qualities[bestIdx]) { BestKnownQualityParameter.ActualValue = new DoubleValue(qualities[bestIdx]); BestKnownSolutionParameter.ActualValue = newBestSolution; } } } #region Problem instance handling public void Load(BPPData data) { BestKnownSolutionParameter.Value = null; BestKnownQualityParameter.Value = null; if (data.BestKnownQuality.HasValue) BestKnownQuality = data.BestKnownQuality.Value; BinShape = data.BinShape; var items = new ItemList(data.Items); items.Sort((x, y) => y.CompareTo(x)); Items = items.AsReadOnly(); ApplyHorizontalOrientation(); LowerBoundParameter.Value.Value = CalculateLowerBound(); } public BPPData Export() { return new BPPData { Name = Name, Description = Description, BinShape = BinShape, Items = Items.ToArray() }; } #endregion #region helpers private void ApplyHorizontalOrientation() { BinShape.ApplyHorizontalOrientation(); foreach (var shape in Items) { shape.ApplyHorizontalOrientation(); } } private int CalculateLowerBound() { //This is the obvious continuous lower bound calculation; Martello and Vigo proposed a better way but it has not been implemented yet; int itemsVol = Items.Select(x => x.Volume).Sum(); int binVol = BinShape.Volume; return (itemsVol + binVol - 1) / (binVol); } #endregion } }