#region License Information
/* HeuristicLab
* Copyright (C) 2002-2010 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 System.Linq;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Data;
using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
using HeuristicLab.Operators;
using HeuristicLab.Optimization;
using HeuristicLab.Parameters;
using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
using HeuristicLab.Problems.DataAnalysis.Symbolic;
using HeuristicLab.Problems.DataAnalysis.Symbolic.Symbols;
namespace HeuristicLab.Problems.DataAnalysis.Regression.Symbolic.Analyzers {
public sealed class SymbolicRegressionTournamentPruning : SingleSuccessorOperator, ISymbolicRegressionAnalyzer {
private const string RandomParameterName = "Random";
private const string SymbolicExpressionTreeParameterName = "SymbolicExpressionTree";
private const string DataAnalysisProblemDataParameterName = "DataAnalysisProblemData";
private const string SamplesStartParameterName = "SamplesStart";
private const string SamplesEndParameterName = "SamplesEnd";
private const string EvaluatorParameterName = "Evaluator";
private const string MaximizationParameterName = "Maximization";
private const string SymbolicExpressionTreeInterpreterParameterName = "SymbolicExpressionTreeInterpreter";
private const string UpperEstimationLimitParameterName = "UpperEstimationLimit";
private const string LowerEstimationLimitParameterName = "LowerEstimationLimit";
private const string MaxPruningRatioParameterName = "MaxPruningRatio";
private const string TournamentSizeParameterName = "TournamentSize";
private const string PopulationPercentileStartParameterName = "PopulationPercentileStart";
private const string PopulationPercentileEndParameterName = "PopulationPercentileEnd";
private const string QualityGainWeightParameterName = "QualityGainWeight";
private const string IterationsParameterName = "Iterations";
private const string FirstPruningGenerationParameterName = "FirstPruningGeneration";
private const string PruningFrequencyParameterName = "PruningFrequency";
private const string GenerationParameterName = "Generations";
private const string ResultsParameterName = "Results";
#region parameter properties
public ILookupParameter RandomParameter {
get { return (ILookupParameter)Parameters[RandomParameterName]; }
}
public ScopeTreeLookupParameter SymbolicExpressionTreeParameter {
get { return (ScopeTreeLookupParameter)Parameters[SymbolicExpressionTreeParameterName]; }
}
public ILookupParameter DataAnalysisProblemDataParameter {
get { return (ILookupParameter)Parameters[DataAnalysisProblemDataParameterName]; }
}
public ILookupParameter SymbolicExpressionTreeInterpreterParameter {
get { return (ILookupParameter)Parameters[SymbolicExpressionTreeInterpreterParameterName]; }
}
public IValueLookupParameter UpperEstimationLimitParameter {
get { return (IValueLookupParameter)Parameters[UpperEstimationLimitParameterName]; }
}
public IValueLookupParameter LowerEstimationLimitParameter {
get { return (IValueLookupParameter)Parameters[LowerEstimationLimitParameterName]; }
}
public IValueLookupParameter SamplesStartParameter {
get { return (IValueLookupParameter)Parameters[SamplesStartParameterName]; }
}
public IValueLookupParameter SamplesEndParameter {
get { return (IValueLookupParameter)Parameters[SamplesEndParameterName]; }
}
public ILookupParameter EvaluatorParameter {
get { return (ILookupParameter)Parameters[EvaluatorParameterName]; }
}
public ILookupParameter MaximizationParameter {
get { return (ILookupParameter)Parameters[MaximizationParameterName]; }
}
public IValueLookupParameter MaxPruningRatioParameter {
get { return (IValueLookupParameter)Parameters[MaxPruningRatioParameterName]; }
}
public IValueLookupParameter TournamentSizeParameter {
get { return (IValueLookupParameter)Parameters[TournamentSizeParameterName]; }
}
public IValueLookupParameter PopulationPercentileStartParameter {
get { return (IValueLookupParameter)Parameters[PopulationPercentileStartParameterName]; }
}
public IValueLookupParameter PopulationPercentileEndParameter {
get { return (IValueLookupParameter)Parameters[PopulationPercentileEndParameterName]; }
}
public IValueLookupParameter QualityGainWeightParameter {
get { return (IValueLookupParameter)Parameters[QualityGainWeightParameterName]; }
}
public IValueLookupParameter IterationsParameter {
get { return (IValueLookupParameter)Parameters[IterationsParameterName]; }
}
public IValueLookupParameter FirstPruningGenerationParameter {
get { return (IValueLookupParameter)Parameters[FirstPruningGenerationParameterName]; }
}
public IValueLookupParameter PruningFrequencyParameter {
get { return (IValueLookupParameter)Parameters[PruningFrequencyParameterName]; }
}
public ILookupParameter GenerationParameter {
get { return (ILookupParameter)Parameters[GenerationParameterName]; }
}
public ILookupParameter ResultsParameter {
get { return (ILookupParameter)Parameters[ResultsParameterName]; }
}
#endregion
#region properties
public IRandom Random {
get { return RandomParameter.ActualValue; }
}
public ItemArray SymbolicExpressionTree {
get { return SymbolicExpressionTreeParameter.ActualValue; }
}
public DataAnalysisProblemData DataAnalysisProblemData {
get { return DataAnalysisProblemDataParameter.ActualValue; }
}
public ISymbolicExpressionTreeInterpreter SymbolicExpressionTreeInterpreter {
get { return SymbolicExpressionTreeInterpreterParameter.ActualValue; }
}
public DoubleValue UpperEstimationLimit {
get { return UpperEstimationLimitParameter.ActualValue; }
}
public DoubleValue LowerEstimationLimit {
get { return LowerEstimationLimitParameter.ActualValue; }
}
public IntValue SamplesStart {
get { return SamplesStartParameter.ActualValue; }
}
public IntValue SamplesEnd {
get { return SamplesEndParameter.ActualValue; }
}
public ISymbolicRegressionEvaluator Evaluator {
get { return EvaluatorParameter.ActualValue; }
}
public BoolValue Maximization {
get { return MaximizationParameter.ActualValue; }
}
public DoubleValue MaxPruningRatio {
get { return MaxPruningRatioParameter.ActualValue; }
}
public IntValue TournamentSize {
get { return TournamentSizeParameter.ActualValue; }
}
public DoubleValue PopulationPercentileStart {
get { return PopulationPercentileStartParameter.ActualValue; }
}
public DoubleValue PopulationPercentileEnd {
get { return PopulationPercentileEndParameter.ActualValue; }
}
public DoubleValue QualityGainWeight {
get { return QualityGainWeightParameter.ActualValue; }
}
public IntValue Iterations {
get { return IterationsParameter.ActualValue; }
}
public IntValue PruningFrequency {
get { return PruningFrequencyParameter.ActualValue; }
}
public IntValue FirstPruningGeneration {
get { return FirstPruningGenerationParameter.ActualValue; }
}
public IntValue Generation {
get { return GenerationParameter.ActualValue; }
}
#endregion
[StorableConstructor]
private SymbolicRegressionTournamentPruning(bool deserializing) : base(deserializing) { }
private SymbolicRegressionTournamentPruning(SymbolicRegressionTournamentPruning original, Cloner cloner) : base(original, cloner) { }
public SymbolicRegressionTournamentPruning()
: base() {
Parameters.Add(new LookupParameter(RandomParameterName, "A random number generator."));
Parameters.Add(new ScopeTreeLookupParameter(SymbolicExpressionTreeParameterName, "The symbolic expression trees to prune."));
Parameters.Add(new LookupParameter(DataAnalysisProblemDataParameterName, "The data analysis problem data to use for branch impact evaluation."));
Parameters.Add(new LookupParameter(SymbolicExpressionTreeInterpreterParameterName, "The interpreter to use for node impact evaluation"));
Parameters.Add(new ValueLookupParameter(SamplesStartParameterName, "The first row index of the dataset partition to use for branch impact evaluation."));
Parameters.Add(new ValueLookupParameter(SamplesEndParameterName, "The last row index of the dataset partition to use for branch impact evaluation."));
Parameters.Add(new LookupParameter(EvaluatorParameterName, "The evaluator that should be used to determine which branches are not relevant."));
Parameters.Add(new LookupParameter(MaximizationParameterName, "The direction of optimization."));
Parameters.Add(new ValueLookupParameter(MaxPruningRatioParameterName, "The maximal relative size of the pruned branch.", new DoubleValue(0.5)));
Parameters.Add(new ValueLookupParameter(TournamentSizeParameterName, "The number of branches to compare for pruning", new IntValue(10)));
Parameters.Add(new ValueLookupParameter(PopulationPercentileStartParameterName, "The start of the population percentile to consider for pruning.", new DoubleValue(0.25)));
Parameters.Add(new ValueLookupParameter(PopulationPercentileEndParameterName, "The end of the population percentile to consider for pruning.", new DoubleValue(0.75)));
Parameters.Add(new ValueLookupParameter(QualityGainWeightParameterName, "The weight of the quality gain relative to the size gain.", new DoubleValue(1.0)));
Parameters.Add(new ValueLookupParameter(UpperEstimationLimitParameterName, "The upper estimation limit to use for evaluation."));
Parameters.Add(new ValueLookupParameter(LowerEstimationLimitParameterName, "The lower estimation limit to use for evaluation."));
Parameters.Add(new ValueLookupParameter(IterationsParameterName, "The number of pruning iterations to apply for each tree.", new IntValue(1)));
Parameters.Add(new ValueLookupParameter(FirstPruningGenerationParameterName, "The first generation when pruning should be applied.", new IntValue(1)));
Parameters.Add(new ValueLookupParameter(PruningFrequencyParameterName, "The frequency of pruning operations (1: every generation, 2: every second generation...)", new IntValue(1)));
Parameters.Add(new LookupParameter(GenerationParameterName, "The current generation."));
Parameters.Add(new LookupParameter(ResultsParameterName, "The results collection."));
}
public override IDeepCloneable Clone(Cloner cloner) {
return new SymbolicRegressionTournamentPruning(this, cloner);
}
[StorableHook(HookType.AfterDeserialization)]
private void AfterDeserialization() {
#region compatibility remove before releasing 3.3.1
if (!Parameters.ContainsKey(EvaluatorParameterName)) {
Parameters.Add(new LookupParameter(EvaluatorParameterName, "The evaluator which should be used to evaluate the solution on the validation set."));
}
if (!Parameters.ContainsKey(MaximizationParameterName)) {
Parameters.Add(new LookupParameter(MaximizationParameterName, "The direction of optimization."));
}
#endregion
}
public override IOperation Apply() {
bool pruningCondition =
(Generation.Value >= FirstPruningGeneration.Value) &&
((Generation.Value - FirstPruningGeneration.Value) % PruningFrequency.Value == 0);
if (pruningCondition) {
int n = SymbolicExpressionTree.Length;
double percentileStart = PopulationPercentileStart.Value;
double percentileEnd = PopulationPercentileEnd.Value;
// for each tree in the given percentile
var trees = SymbolicExpressionTree
.Skip((int)(n * percentileStart))
.Take((int)(n * (percentileEnd - percentileStart)));
foreach (var tree in trees) {
Prune(Random, tree, Iterations.Value, TournamentSize.Value,
DataAnalysisProblemData, SamplesStart.Value, SamplesEnd.Value,
SymbolicExpressionTreeInterpreter, Evaluator, Maximization.Value,
LowerEstimationLimit.Value, UpperEstimationLimit.Value,
MaxPruningRatio.Value, QualityGainWeight.Value);
}
}
return base.Apply();
}
public static void Prune(IRandom random, SymbolicExpressionTree tree, int iterations, int tournamentSize,
DataAnalysisProblemData problemData, int samplesStart, int samplesEnd,
ISymbolicExpressionTreeInterpreter interpreter, ISymbolicRegressionEvaluator evaluator, bool maximization,
double lowerEstimationLimit, double upperEstimationLimit,
double maxPruningRatio, double qualityGainWeight) {
IEnumerable rows = Enumerable.Range(samplesStart, samplesEnd - samplesStart)
.Where(i => i < problemData.TestSamplesStart.Value || problemData.TestSamplesEnd.Value <= i);
int originalSize = tree.Size;
double originalQuality = evaluator.Evaluate(interpreter, tree,
lowerEstimationLimit, upperEstimationLimit, problemData.Dataset, problemData.TargetVariable.Value, rows);
int minPrunedSize = (int)(originalSize * (1 - maxPruningRatio));
// tree for branch evaluation
SymbolicExpressionTree templateTree = (SymbolicExpressionTree)tree.Clone();
while (templateTree.Root.SubTrees[0].SubTrees.Count > 0) templateTree.Root.SubTrees[0].RemoveSubTree(0);
SymbolicExpressionTree prunedTree = tree;
for (int iteration = 0; iteration < iterations; iteration++) {
SymbolicExpressionTree iterationBestTree = prunedTree;
double bestGain = double.PositiveInfinity;
int maxPrunedBranchSize = (int)(prunedTree.Size * maxPruningRatio);
for (int i = 0; i < tournamentSize; i++) {
var clonedTree = (SymbolicExpressionTree)prunedTree.Clone();
int clonedTreeSize = clonedTree.Size;
var prunePoints = (from node in clonedTree.IterateNodesPostfix()
from subTree in node.SubTrees
let subTreeSize = subTree.GetSize()
where subTreeSize <= maxPrunedBranchSize
where clonedTreeSize - subTreeSize >= minPrunedSize
select new { Parent = node, Branch = subTree, SubTreeIndex = node.SubTrees.IndexOf(subTree) })
.ToList();
if (prunePoints.Count > 0) {
var selectedPrunePoint = prunePoints.SelectRandom(random);
templateTree.Root.SubTrees[0].AddSubTree(selectedPrunePoint.Branch);
IEnumerable branchValues = interpreter.GetSymbolicExpressionTreeValues(templateTree, problemData.Dataset, rows);
double branchMean = branchValues.Average();
templateTree.Root.SubTrees[0].RemoveSubTree(0);
selectedPrunePoint.Parent.RemoveSubTree(selectedPrunePoint.SubTreeIndex);
var constNode = CreateConstant(branchMean);
selectedPrunePoint.Parent.InsertSubTree(selectedPrunePoint.SubTreeIndex, constNode);
double prunedQuality = evaluator.Evaluate(interpreter, clonedTree,
lowerEstimationLimit, upperEstimationLimit, problemData.Dataset, problemData.TargetVariable.Value, Enumerable.Range(samplesStart, samplesEnd - samplesStart));
double prunedSize = clonedTree.Size;
// deteriation in quality:
// exp: MSE : newMse < origMse (improvement) => prefer the larger improvement
// MSE : newMse > origMse (deteriation) => prefer the smaller deteriation
// MSE : minimize: newMse / origMse
// R˛ : newR˛ > origR˛ (improvment) => prefer the larger improvment
// R˛ : newR˛ < origR˛ (deteriation) => prefer smaller deteriation
// R˛ : minimize: origR˛ / newR˛
double qualityDeteriation = maximization ? originalQuality / prunedQuality : prunedQuality / originalQuality;
// size of the pruned tree is always smaller than the size of the original tree
// same change in quality => prefer pruning operation that removes a larger tree
double gain = (qualityDeteriation * qualityGainWeight) /
(originalSize / prunedSize);
if (gain < bestGain) {
bestGain = gain;
iterationBestTree = clonedTree;
}
}
}
prunedTree = iterationBestTree;
}
tree.Root = prunedTree.Root;
}
private static SymbolicExpressionTreeNode CreateConstant(double constantValue) {
var node = (ConstantTreeNode)(new Constant()).CreateTreeNode();
node.Value = constantValue;
return node;
}
}
}