#region License Information /* HeuristicLab * Copyright (C) 2002-2015 Joseph Helm and 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.Problems.BinPacking.Interfaces; using HeuristicLab.Problems.BinPacking.Shapes; using HeuristicLab.Core; using HeuristicLab.Persistence.Default.CompositeSerializers.Storable; using HeuristicLab.Common; using HeuristicLab.Parameters; using HeuristicLab.Problems.BinPacking.Decoders; using HeuristicLab.PluginInfrastructure; using HeuristicLab.Data; using HeuristicLab.Problems.BinPacking.Analyzers; using HeuristicLab.Encodings.PackingEncoding.PackingSequence; using HeuristicLab.Encodings.PackingEncoding.GroupingVector; using HeuristicLab.Problems.Instances; using HeuristicLab.Encodings.PackingEncoding.MultiComponentVector; namespace HeuristicLab.Problems.BinPacking.Problem { [Item("RegularIdenticalBinPackingProblem", "Represents a bin-packing problem-instance using only bins with identical measures and bins/items with regular shapes (e.g. rectangle, cuboid).")] [StorableClass] public abstract class RegularIdenticalBinPackingProblem : BinPackingProblem, IProblemInstanceConsumer, IProblemInstanceExporter where D : class, IPackingDimensions where B : PackingShape, IPackingBin, IRegularPackingShape, new() where I : PackingShape, IPackingItem, IRegularPackingShape, new() { #region Parameter Properties public ValueParameter PackingBinMeasuresParameter { get { return (ValueParameter)Parameters["PackingBinMeasures"]; } } public ConstrainedValueParameter PackingSolutionDecoderParameter { get { return (ConstrainedValueParameter)Parameters["PackingSolutionDecoder"]; } } #endregion #region Properties public B PackingBinMeasures { get { return PackingBinMeasuresParameter.Value; } set { PackingBinMeasuresParameter.Value = value; } } public PackingSolutionDecoder PackingSolutionDecoder { get { return PackingSolutionDecoderParameter.Value as PackingSolutionDecoder; } set { PackingSolutionDecoderParameter.Value = value; } } #endregion [StorableConstructor] protected RegularIdenticalBinPackingProblem(bool deserializing) : base(deserializing) { } protected RegularIdenticalBinPackingProblem(RegularIdenticalBinPackingProblem original, Cloner cloner) : base(original, cloner) { InitializeEventHandlers(); } protected RegularIdenticalBinPackingProblem(IPackingPlanEvaluationAlgorithm e) : this(e, new MultiComponentVectorRandomCreator()) { } protected RegularIdenticalBinPackingProblem(IPackingPlanEvaluationAlgorithm e, IPackingSolutionCreator c) : base(e, c) { Parameters.Add(new ValueParameter("PackingBinMeasures", "Packing-bin data defining the measures of the used bins.", new B())); Parameters.Add(new ValueParameter("PackingPlanEvaluator", "The evaluator is used to determine the quality of a solution.", CreateDefaultEvaluator())); Parameters.Add(new ConstrainedValueParameter("PackingSolutionDecoder", "The operator that decodes the representation and creates a packing plan.")); this.Maximization.Value = true; InitializeProblemInstance(); InitializeOperators(); InitializeEventHandlers(); } #region Helpers protected override void InitializeProblemInstance() { InitializeProblemData(); ApplyHorizontalOrientation(); SortItems(); RemoveTooBigItems(); PackingItemsParameter.Value.Value = PackingItemMeasures.Count; LowerBoundParameter.Value.Value = CalculateLowerBound(); } protected abstract void RemoveTooBigItems(); protected abstract void InitializeProblemData(); protected abstract IPackingPlanEvaluator CreateDefaultEvaluator(); private void ApplyHorizontalOrientation() { PackingBinMeasures.ApplyHorizontalOrientation(); foreach (IRegularPackingShape shape in PackingItemMeasures) { shape.ApplyHorizontalOrientation(); } } private void SortItems() { PackingItemMeasures.Sort((x, y) => y.CompareTo(x)); } 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 items = PackingItemMeasures.Select(x => x.MultipliedMeasures).Sum(); int bin = PackingBinMeasures.MultipliedMeasures; return (items + bin - 1)/(bin); } protected override void InitializeOperators() { Operators.Clear(); Operators.Add(new BestBinPackingSolutionAnalyzer()); ApplyEncoding(); ParameterizeOperators(); } private void ApplyEncoding() { if (SolutionCreator.GetType() == typeof(PackingSequenceRandomCreator)) { Operators.AddRange(ApplicationManager.Manager.GetInstances()); InitializeDecoder(); } else if (SolutionCreator.GetType() == typeof(GroupingVectorRandomCreator)) { Operators.AddRange(ApplicationManager.Manager.GetInstances()); InitializeDecoder(); } else if (SolutionCreator.GetType() == typeof(MultiComponentVectorRandomCreator)) { Operators.AddRange(ApplicationManager.Manager.GetInstances()); InitializeDecoder(); } } private void ParameterizeOperators() { foreach (var op in Operators.OfType>()) { op.QualityParameter.ActualName = PackingPlanEvaluator.QualityParameter.ActualName; } Evaluator.PackingSolutionDecoderParameter.ActualName = PackingSolutionDecoderParameter.Name; Evaluator.PackingSolutionDecoderParameter.Hidden = true; Evaluator.PackingPlanEvaluatorParameter.ActualName = PackingPlanEvaluatorParameter.Name; Evaluator.PackingPlanEvaluatorParameter.Hidden = true; Evaluator.QualityParameter.ActualName = PackingPlanEvaluator.QualityParameter.ActualName; Evaluator.QualityParameter.Hidden = true; if (PackingSolutionDecoder != null) PackingSolutionDecoder.EncodedSolutionParameter.ActualName = SolutionCreator.EncodedSolutionParameter.ActualName; if (PackingSolutionDecoder != null) { PackingPlanEvaluator.PackingPlanParameter.ActualName = PackingSolutionDecoder.PackingPlanParameter.ActualName; PackingPlanEvaluator.PackingPlanParameter.Hidden = true; } else { PackingPlanEvaluator.PackingPlanParameter.ActualName = PackingPlanEvaluator.PackingPlanParameter.Name; PackingPlanEvaluator.PackingPlanParameter.Hidden = false; } foreach (var op in Operators.OfType()) { op.EncodedSolutionParameter.ActualName = SolutionCreator.EncodedSolutionParameter.ActualName; op.EncodedSolutionParameter.Hidden = true; } foreach (var op in Operators.OfType()) { op.ChildParameter.ActualName = SolutionCreator.EncodedSolutionParameter.ActualName; op.ChildParameter.Hidden = true; op.ParentsParameter.ActualName = SolutionCreator.EncodedSolutionParameter.ActualName; op.ParentsParameter.Hidden = true; } foreach (var op in Operators.OfType>()) { op.QualityParameter.ActualName = PackingPlanEvaluator.QualityParameter.ActualName; if (PackingSolutionDecoder != null) { op.PackingPlanParameter.ActualName = PackingSolutionDecoder.PackingPlanParameter.ActualName; op.PackingPlanParameter.Hidden = true; } else { op.PackingPlanParameter.ActualName = op.PackingPlanParameter.Name; op.PackingPlanParameter.Hidden = false; } } } protected override void InitializeEventHandlers() { PackingPlanEvaluatorParameter.ValueChanged += PackingPlanEvaluatorParameter_ValueChanged; PackingPlanEvaluator.QualityParameter.ActualNameChanged += PackingPlanEvaluator_QualityParameter_ActualNameChanged; SolutionCreatorParameter.ValueChanged += SolutionCreatorParameter_ValueChanged; SolutionCreator.EncodedSolutionParameter.ActualNameChanged += SolutionCreator_EncodedSolutionParameter_ActualNameChanged; PackingSolutionDecoderParameter.ValueChanged += PackingSolutionDecoderParameter_ValueChanged; if (PackingSolutionDecoder != null) PackingSolutionDecoder.PackingPlanParameter.ActualNameChanged += PackingSolutionDecoder_PackingPlanParameter_ActualNameChanged; } #endregion #region Events protected override void OnSolutionCreatorChanged() { InitializeOperators(); } protected override void OnEvaluatorChanged() { base.OnEvaluatorChanged(); ParameterizeOperators(); } private void PackingPlanEvaluatorParameter_ValueChanged(object sender, EventArgs eventArgs) { PackingPlanEvaluator.QualityParameter.ActualNameChanged += PackingPlanEvaluator_QualityParameter_ActualNameChanged; ParameterizeOperators(); } private void PackingPlanEvaluator_QualityParameter_ActualNameChanged(object sender, EventArgs eventArgs) { ParameterizeOperators(); } private void SolutionCreatorParameter_ValueChanged(object sender, EventArgs eventArgs) { SolutionCreator.EncodedSolutionParameter.ActualNameChanged += SolutionCreator_EncodedSolutionParameter_ActualNameChanged; ParameterizeOperators(); } private void SolutionCreator_EncodedSolutionParameter_ActualNameChanged(object sender, EventArgs eventArgs) { ParameterizeOperators(); } private void PackingSolutionDecoderParameter_ValueChanged(object sender, EventArgs eventArgs) { if (PackingSolutionDecoder != null) PackingSolutionDecoder.PackingPlanParameter.ActualNameChanged += PackingSolutionDecoder_PackingPlanParameter_ActualNameChanged; ParameterizeOperators(); } private void PackingSolutionDecoder_PackingPlanParameter_ActualNameChanged(object sender, EventArgs eventArgs) { ParameterizeOperators(); } #endregion #region Problem instance handling public virtual void Load(BPPData data) { var realData = data as RealBPPData; var binData = new B(); binData.InitializeFromMeasures (data.BinMeasures); var itemData = new ItemList(data.Items); for (int j = 0; j < data.Items; j++) { var item = new I(); item.InitializeFromMeasures(data.ItemMeasures[j]); item.AddTargetBinMeasures(data.BinMeasures); if (realData != null) { item.Weight = realData.ItemWeights[j]; item.Material = realData.ItemMaterials[j]; } itemData.Add(item); } BestKnownQuality = data.BestKnownQuality.HasValue ? new DoubleValue(data.BestKnownQuality.Value) : null; PackingBinMeasures = binData; //PackingItemsParameter.Value.Value = data.Items; PackingItemMeasures = itemData; ApplyHorizontalOrientation(); SortItems(); PackingItemsParameter.Value.Value = PackingItemMeasures.Count; LowerBoundParameter.Value.Value = CalculateLowerBound(); } public BPPData Export() { var result = new BPPData{ Name = Name, Description = Description, Items = PackingItemsParameter.Value.Value, BinMeasures = PackingBinMeasures.ToArray() }; var itemMeasures = new int[result.Items][]; int i = 0; foreach (var item in PackingItemMeasures) { itemMeasures[i] = item.ToArray(); i++; } result.ItemMeasures = itemMeasures; return result; } #endregion } }