Free cookie consent management tool by TermsFeed Policy Generator

source: branches/3136_Structural_GP/HeuristicLab.Problems.DataAnalysis.Symbolic.Regression/3.4/SingleObjective/Evaluators/NMSESingleObjectiveConstraintsEvaluator.cs @ 18104

Last change on this file since 18104 was 18104, checked in by dpiringe, 2 years ago

#3136

  • overrode the method GetActualValue in ValueLookupParameter to get the default value when the execution context is null
  • reverted the linear scaling logic for NMSESingleObjectiveConstraintsEvaluator
  • in SymbolicRegressionConstantOptimizationEvaluator: removed the usage of GenerateRowsToEvaluate because it uses lookup parameters
  • set the value of RelativeNumberOfEvaluatedSamplesParameter for SymbolicRegressionConstantOptimizationEvaluator in StructuredSymbolicRegressionSingleObjectiveProblem if Maximization = true and the SymbolicRegressionConstantOptimizationEvaluator is configured as evaluator
  • added the SubFunctionSymbol in TreeToAutoDiffTermConverter
File size: 11.9 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) Heuristic and Evolutionary Algorithms Laboratory (HEAL)
4 *
5 * This file is part of HeuristicLab.
6 *
7 * HeuristicLab is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * HeuristicLab is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
19 */
20#endregion
21
22using System;
23using System.Collections.Generic;
24using System.Linq;
25using HEAL.Attic;
26using HeuristicLab.Common;
27using HeuristicLab.Core;
28using HeuristicLab.Data;
29using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
30using HeuristicLab.Parameters;
31
32namespace HeuristicLab.Problems.DataAnalysis.Symbolic.Regression {
33  [Item("NMSE Evaluator with shape-constraints (single-objective)", "Calculates NMSE of a symbolic regression solution and checks constraints. The fitness is a combination of NMSE and constraint violations.")]
34  [StorableType("27473973-DD8D-4375-997D-942E2280AE8E")]
35  public class NMSESingleObjectiveConstraintsEvaluator : SymbolicRegressionSingleObjectiveEvaluator {
36    #region Parameter/Properties
37
38    private const string OptimizeParametersParameterName = "OptimizeParameters";
39    private const string ParameterOptimizationIterationsParameterName = "ParameterOptimizationIterations";
40    private const string UseSoftConstraintsParameterName = "UseSoftConstraintsEvaluation";
41    private const string BoundsEstimatorParameterName = "BoundsEstimator";
42    private const string PenaltyFactorParameterName = "PenaltyFactor";
43
44
45    public IFixedValueParameter<BoolValue> OptimizerParametersParameter =>
46      (IFixedValueParameter<BoolValue>)Parameters[OptimizeParametersParameterName];
47
48    public IFixedValueParameter<IntValue> ConstantOptimizationIterationsParameter =>
49      (IFixedValueParameter<IntValue>)Parameters[ParameterOptimizationIterationsParameterName];
50
51    public IFixedValueParameter<BoolValue> UseSoftConstraintsParameter =>
52      (IFixedValueParameter<BoolValue>)Parameters[UseSoftConstraintsParameterName];
53
54    public IValueParameter<IBoundsEstimator> BoundsEstimatorParameter =>
55      (IValueParameter<IBoundsEstimator>)Parameters[BoundsEstimatorParameterName];
56    public IFixedValueParameter<DoubleValue> PenaltyFactorParameter =>
57      (IFixedValueParameter<DoubleValue>)Parameters[PenaltyFactorParameterName];
58
59    public bool OptimizeParameters {
60      get => OptimizerParametersParameter.Value.Value;
61      set => OptimizerParametersParameter.Value.Value = value;
62    }
63
64    public int ConstantOptimizationIterations {
65      get => ConstantOptimizationIterationsParameter.Value.Value;
66      set => ConstantOptimizationIterationsParameter.Value.Value = value;
67    }
68
69    public bool UseSoftConstraints {
70      get => UseSoftConstraintsParameter.Value.Value;
71      set => UseSoftConstraintsParameter.Value.Value = value;
72    }
73
74    public IBoundsEstimator BoundsEstimator {
75      get => BoundsEstimatorParameter.Value;
76      set => BoundsEstimatorParameter.Value = value;
77    }
78
79    public double PenalityFactor {
80      get => PenaltyFactorParameter.Value.Value;
81      set => PenaltyFactorParameter.Value.Value = value;
82    }
83
84
85    public override bool Maximization => false; // NMSE is minimized
86
87    #endregion
88
89    #region Constructors/Cloning
90
91    [StorableConstructor]
92    protected NMSESingleObjectiveConstraintsEvaluator(StorableConstructorFlag _) : base(_) { }
93
94    protected NMSESingleObjectiveConstraintsEvaluator(
95      NMSESingleObjectiveConstraintsEvaluator original, Cloner cloner) : base(original, cloner) { }
96
97    public NMSESingleObjectiveConstraintsEvaluator() {
98      Parameters.Add(new FixedValueParameter<BoolValue>(OptimizeParametersParameterName,
99        "Define whether optimization of numeric parameters is active or not (default: false).", new BoolValue(false)));
100      Parameters.Add(new FixedValueParameter<IntValue>(ParameterOptimizationIterationsParameterName,
101        "Define how many parameter optimization steps should be performed (default: 10).", new IntValue(10)));
102      Parameters.Add(new FixedValueParameter<BoolValue>(UseSoftConstraintsParameterName,
103        "Define whether the constraints are penalized by soft or hard constraints (default: false).", new BoolValue(false)));
104      Parameters.Add(new ValueParameter<IBoundsEstimator>(BoundsEstimatorParameterName,
105        "The estimator which is used to estimate output ranges of models (default: interval arithmetic).", new IntervalArithBoundsEstimator()));
106      Parameters.Add(new FixedValueParameter<DoubleValue>(PenaltyFactorParameterName,
107        "Punishment factor for constraint violations for soft constraint handling (fitness = NMSE + penaltyFactor * avg(violations)) (default: 1.0)", new DoubleValue(1.0)));
108    }
109
110    [StorableHook(HookType.AfterDeserialization)]
111    private void AfterDeserialization() { }
112
113    public override IDeepCloneable Clone(Cloner cloner) {
114      return new NMSESingleObjectiveConstraintsEvaluator(this, cloner);
115    }
116
117    #endregion
118
119    public override IOperation InstrumentedApply() {
120      var rows = GenerateRowsToEvaluate();
121      var tree = SymbolicExpressionTreeParameter.ActualValue;
122      var problemData = ProblemDataParameter.ActualValue;
123      var interpreter = SymbolicDataAnalysisTreeInterpreterParameter.ActualValue;
124      var estimationLimits = EstimationLimitsParameter.ActualValue;
125      var applyLinearScaling = ApplyLinearScalingParameter.ActualValue.Value;
126
127      var quality = Evaluate(tree, problemData, rows, interpreter, applyLinearScaling, estimationLimits.Lower, estimationLimits.Upper);
128      QualityParameter.ActualValue = new DoubleValue(quality);
129
130      return base.InstrumentedApply();
131    }
132
133    private static void CalcLinearScalingTerms(
134      ISymbolicExpressionTree tree,
135      IRegressionProblemData problemData,
136      IEnumerable<int> rows,
137      ISymbolicDataAnalysisExpressionTreeInterpreter interpreter) {
138      var rootNode = new ProgramRootSymbol().CreateTreeNode();
139      var startNode = new StartSymbol().CreateTreeNode();
140      var offset = tree.Root.GetSubtree(0) //Start
141                            .GetSubtree(0); //Offset
142      var scaling = offset.GetSubtree(0);
143     
144      //Check if tree contains offset and scaling nodes
145      if (!(offset.Symbol is Addition) || !(scaling.Symbol is Multiplication))
146        throw new ArgumentException($"Shape Constraints Evaluation can only be used with LinearScalingGrammar.");
147
148      var t = (ISymbolicExpressionTreeNode)scaling.GetSubtree(0).Clone();
149      rootNode.AddSubtree(startNode);
150      startNode.AddSubtree(t);
151      var newTree = new SymbolicExpressionTree(rootNode);
152
153      //calculate alpha and beta for scaling
154      var estimatedValues = interpreter.GetSymbolicExpressionTreeValues(newTree, problemData.Dataset, rows);
155
156      var targetValues = problemData.Dataset.GetDoubleValues(problemData.TargetVariable, rows);
157      OnlineLinearScalingParameterCalculator.Calculate(estimatedValues, targetValues, out var alpha, out var beta, out var errorState);
158      if (errorState == OnlineCalculatorError.None) {
159        //Set alpha and beta to the scaling nodes from ia grammar
160        var offsetParameter = offset.GetSubtree(1) as ConstantTreeNode;
161        offsetParameter.Value = alpha;
162        var scalingParameter = scaling.GetSubtree(1) as ConstantTreeNode;
163        scalingParameter.Value = beta;
164      }
165    }
166
167    public static double Calculate(
168      ISymbolicExpressionTree tree,
169      IRegressionProblemData problemData, IEnumerable<int> rows,
170      ISymbolicDataAnalysisExpressionTreeInterpreter interpreter,
171      double lowerEstimationLimit, double upperEstimationLimit,
172      IBoundsEstimator estimator,
173      bool useSoftConstraints = false, double penaltyFactor = 1.0) {
174
175      var constraints = Enumerable.Empty<ShapeConstraint>();
176      if (problemData is ShapeConstrainedRegressionProblemData scProbData)
177        constraints = scProbData.ShapeConstraints.EnabledConstraints;
178
179      var estimatedValues = interpreter.GetSymbolicExpressionTreeValues(tree, problemData.Dataset, rows);
180      var boundedEstimatedValues = estimatedValues.LimitToRange(lowerEstimationLimit, upperEstimationLimit);
181      var targetValues = problemData.Dataset.GetDoubleValues(problemData.TargetVariable, rows);
182      var nmse = OnlineNormalizedMeanSquaredErrorCalculator.Calculate(targetValues, boundedEstimatedValues, out var errorState);
183      if (errorState != OnlineCalculatorError.None)
184        return 1.0;
185
186      if (!constraints.Any())
187        return nmse;
188
189      var intervalCollection = problemData.VariableRanges;
190      var constraintViolations = IntervalUtil.GetConstraintViolations(constraints, estimator, intervalCollection, tree);
191
192      // infinite/NaN constraints
193      if (constraintViolations.Any(x => double.IsNaN(x) || double.IsInfinity(x)))
194        return 1.0;
195     
196      // hard constraints
197      if (!useSoftConstraints) {
198        if (constraintViolations.Any(x => x > 0.0))
199          return 1.0;
200        return nmse;
201      }
202
203      // soft constraints
204      if (penaltyFactor < 0.0)
205        throw new ArgumentException("The parameter has to be >= 0.0.", nameof(penaltyFactor));
206
207      var weightedViolationsAvg = constraints
208        .Zip(constraintViolations, (c, v) => c.Weight * v)
209        .Average();
210
211      return Math.Min(nmse, 1.0) + penaltyFactor * weightedViolationsAvg;
212    }
213
214    public override double Evaluate(
215      IExecutionContext context, ISymbolicExpressionTree tree, IRegressionProblemData problemData,
216      IEnumerable<int> rows) {
217      SymbolicDataAnalysisTreeInterpreterParameter.ExecutionContext = context;
218      EstimationLimitsParameter.ExecutionContext = context;
219      ApplyLinearScalingParameter.ExecutionContext = context;
220
221      var nmse = Calculate(
222        tree, problemData, rows,
223        SymbolicDataAnalysisTreeInterpreterParameter.ActualValue,
224        EstimationLimitsParameter.ActualValue.Lower,
225        EstimationLimitsParameter.ActualValue.Upper,
226        BoundsEstimator,
227        UseSoftConstraints,
228        PenalityFactor);
229
230      SymbolicDataAnalysisTreeInterpreterParameter.ExecutionContext = null;
231      EstimationLimitsParameter.ExecutionContext = null;
232      ApplyLinearScalingParameter.ExecutionContext = null;
233
234      return nmse;
235    }
236
237    public override double Evaluate(
238      ISymbolicExpressionTree tree,
239      IRegressionProblemData problemData,
240      IEnumerable<int> rows,
241      ISymbolicDataAnalysisExpressionTreeInterpreter interpreter,
242      bool applyLinearScaling = true,
243      double lowerEstimationLimit = double.MinValue,
244      double upperEstimationLimit = double.MaxValue) {
245
246      if (OptimizeParameters)
247        SymbolicRegressionConstantOptimizationEvaluator.OptimizeConstants(
248          interpreter, tree,
249          problemData, rows,
250          applyLinearScaling: false, // Tree already contains scaling terms
251          ConstantOptimizationIterations,
252          updateVariableWeights: true,
253          lowerEstimationLimit,
254          upperEstimationLimit);
255     
256      else if (applyLinearScaling) // extra scaling terms, which are included in tree
257        CalcLinearScalingTerms(tree, problemData, rows, interpreter);
258     
259      return Calculate(
260        tree, problemData,
261        rows, interpreter,
262        lowerEstimationLimit,
263        upperEstimationLimit,
264        BoundsEstimator,
265        UseSoftConstraints,
266        PenalityFactor);
267    }
268  }
269}
Note: See TracBrowser for help on using the repository browser.