#region License Information
/* HeuristicLab
* Copyright (C) 2002-2019 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.Collections.Generic;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Encodings.BinaryVectorEncoding;
using HeuristicLab.Parameters;
using HEAL.Attic;
namespace HeuristicLab.Problems.NK {
[Item("NK BitFlip Move Evaluator", "Evaluates a single bit flip on an NK landscape.")]
[StorableType("58DA2DE8-6FEB-464F-9B25-E2CA676D530E")]
public class NKBitFlipMoveEvaluator : NKMoveEvaluator, IOneBitflipMoveOperator {
public ILookupParameter OneBitflipMoveParameter {
get { return (ILookupParameter)Parameters["OneBitflipMove"]; }
}
public ILookupParameter MovedBinaryVectorParameter {
get { return (ILookupParameter)Parameters["MovedBinaryVector"]; }
}
[StorableConstructor]
protected NKBitFlipMoveEvaluator(StorableConstructorFlag _) : base(_) { }
protected NKBitFlipMoveEvaluator(NKBitFlipMoveEvaluator original, Cloner cloner)
: base(original, cloner) { }
public NKBitFlipMoveEvaluator()
: base() {
Parameters.Add(new LookupParameter("OneBitflipMove", "The move to evaluate."));
Parameters.Add(new LookupParameter("MovedBinaryVector", "The resulting binary vector after the move."));
}
public override IDeepCloneable Clone(Cloner cloner) {
return new NKBitFlipMoveEvaluator(this, cloner);
}
public override IOperation Apply() {
BinaryVector binaryVector = BinaryVectorParameter.ActualValue;
OneBitflipMove move = OneBitflipMoveParameter.ActualValue;
BoolMatrix interactions = GeneInteractionsParameter.ActualValue;
DoubleArray weights = WeightsParameter.ActualValue;
int seed = InteractionSeedParameter.ActualValue.Value;
double moveQuality = QualityParameter.ActualValue.Value;
int q = QParameter.ActualValue.Value;
double p = PParameter.ActualValue.Value;
List affectedFitnessComponents = new List();
for (int c = 0; c < interactions.Columns; c++)
if (interactions[move.Index, c])
affectedFitnessComponents.Add(c);
BinaryVector moved = new BinaryVector(binaryVector);
MovedBinaryVectorParameter.ActualValue = moved;
moved[move.Index] = !moved[move.Index];
if (affectedFitnessComponents.Count * 2 > interactions.Columns) {
double[] f_i;
moveQuality = NKLandscape.Evaluate(moved, interactions, weights, seed, out f_i, q, p);
} else {
long x = NKLandscape.Encode(binaryVector);
long y = NKLandscape.Encode(moved);
long[] g = NKLandscape.Encode(interactions);
double[] w = NKLandscape.Normalize(weights);
foreach (var c in affectedFitnessComponents) {
moveQuality -= w[c%w.Length]*NKLandscape.F_i(x, c, g[c], seed, q, p);
moveQuality += w[c%w.Length]*NKLandscape.F_i(y, c, g[c], seed, q, p);
}
}
if (MoveQualityParameter.ActualValue == null) MoveQualityParameter.ActualValue = new DoubleValue(moveQuality);
else MoveQualityParameter.ActualValue.Value = moveQuality;
return base.Apply();
}
}
}