Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Interpreter/IntervalArithBoundsEstimator.cs @ 17912

Last change on this file since 17912 was 17911, checked in by gkronber, 4 years ago

#3073 code improvements after reintegration of branch

File size: 13.9 KB
Line 
1#region License Information
2
3/* HeuristicLab
4 * Copyright (C) Heuristic and Evolutionary Algorithms Laboratory (HEAL)
5 *
6 * This file is part of HeuristicLab.
7 *
8 * HeuristicLab is free software: you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation, either version 3 of the License, or
11 * (at your option) any later version.
12 *
13 * HeuristicLab is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22#endregion
23using System;
24using System.Collections.Generic;
25using System.Linq;
26using HEAL.Attic;
27using HeuristicLab.Common;
28using HeuristicLab.Core;
29using HeuristicLab.Data;
30using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
31using HeuristicLab.Parameters;
32
33namespace HeuristicLab.Problems.DataAnalysis.Symbolic {
34  [StorableType("C8539434-6FB0-47D0-9F5A-2CAE5D8B8B4F")]
35  [Item("Interval Arithmetic Bounds Estimator", "Interpreter for calculation of intervals of symbolic models.")]
36  public sealed class IntervalArithBoundsEstimator : ParameterizedNamedItem, IBoundsEstimator {
37    #region Parameters
38
39    private const string EvaluatedSolutionsParameterName = "EvaluatedSolutions";
40
41    public IFixedValueParameter<IntValue> EvaluatedSolutionsParameter =>
42      (IFixedValueParameter<IntValue>)Parameters[EvaluatedSolutionsParameterName];
43
44    public int EvaluatedSolutions {
45      get => EvaluatedSolutionsParameter.Value.Value;
46      set => EvaluatedSolutionsParameter.Value.Value = value;
47    }
48    #endregion
49
50    #region Constructors
51
52    [StorableConstructor]
53    private IntervalArithBoundsEstimator(StorableConstructorFlag _) : base(_) { }
54
55    protected IntervalArithBoundsEstimator(IntervalArithBoundsEstimator original, Cloner cloner) : base(original, cloner) { }
56
57    public IntervalArithBoundsEstimator() : base("Interval Arithmetic Bounds Estimator",
58      "Estimates the bounds of the model with interval arithmetic") {
59      Parameters.Add(new FixedValueParameter<IntValue>(EvaluatedSolutionsParameterName,
60        "A counter for the total number of solutions the estimator has evaluated.", new IntValue(0)));
61    }
62
63    public override IDeepCloneable Clone(Cloner cloner) {
64      return new IntervalArithBoundsEstimator(this, cloner);
65    }
66
67    #endregion
68
69    #region IStatefulItem Members
70
71    private readonly object syncRoot = new object();
72
73    public void InitializeState() {
74      EvaluatedSolutions = 0;
75    }
76
77    public void ClearState() { }
78
79    #endregion
80
81    #region Evaluation
82
83    private static Instruction[] PrepareInterpreterState(
84      ISymbolicExpressionTree tree,
85      IDictionary<string, Interval> variableRanges) {
86      if (variableRanges == null)
87        throw new ArgumentNullException("No variable ranges are present!", nameof(variableRanges));
88
89      // Check if all variables used in the tree are present in the dataset
90      foreach (var variable in tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(n => n.VariableName)
91                                   .Distinct())
92        if (!variableRanges.ContainsKey(variable))
93          throw new InvalidOperationException($"No ranges for variable {variable} is present");
94
95      var code = SymbolicExpressionTreeCompiler.Compile(tree, OpCodes.MapSymbolToOpCode);
96      foreach (var instr in code.Where(i => i.opCode == OpCodes.Variable)) {
97        var variableTreeNode = (VariableTreeNode)instr.dynamicNode;
98        instr.data = variableRanges[variableTreeNode.VariableName];
99      }
100
101      return code;
102    }
103
104    // Use ref parameter, because the tree will be iterated through recursively from the left-side branch to the right side
105    // Update instructionCounter, whenever Evaluate is called
106    public static Interval Evaluate(
107      Instruction[] instructions, ref int instructionCounter,
108      IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null,
109      IDictionary<string, Interval> variableIntervals = null) {
110      var currentInstr = instructions[instructionCounter];
111      instructionCounter++;
112      Interval result;
113
114      switch (currentInstr.opCode) {
115        case OpCodes.Variable: {
116            var variableTreeNode = (VariableTreeNode)currentInstr.dynamicNode;
117            var weightInterval = new Interval(variableTreeNode.Weight, variableTreeNode.Weight);
118
119            Interval variableInterval;
120            if (variableIntervals != null && variableIntervals.ContainsKey(variableTreeNode.VariableName))
121              variableInterval = variableIntervals[variableTreeNode.VariableName];
122            else
123              variableInterval = (Interval)currentInstr.data;
124
125            result = Interval.Multiply(variableInterval, weightInterval);
126            break;
127          }
128        case OpCodes.Constant: {
129            var constTreeNode = (ConstantTreeNode)currentInstr.dynamicNode;
130            result = new Interval(constTreeNode.Value, constTreeNode.Value);
131            break;
132          }
133        case OpCodes.Add: {
134            result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
135            for (var i = 1; i < currentInstr.nArguments; i++) {
136              var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
137              result = Interval.Add(result, argumentInterval);
138            }
139
140            break;
141          }
142        case OpCodes.Sub: {
143            result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
144            if (currentInstr.nArguments == 1)
145              result = Interval.Multiply(new Interval(-1, -1), result);
146
147            for (var i = 1; i < currentInstr.nArguments; i++) {
148              var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
149              result = Interval.Subtract(result, argumentInterval);
150            }
151
152            break;
153          }
154        case OpCodes.Mul: {
155            result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
156            for (var i = 1; i < currentInstr.nArguments; i++) {
157              var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
158              result = Interval.Multiply(result, argumentInterval);
159            }
160
161            break;
162          }
163        case OpCodes.Div: {
164            result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
165            if (currentInstr.nArguments == 1)
166              result = Interval.Divide(new Interval(1, 1), result);
167
168            for (var i = 1; i < currentInstr.nArguments; i++) {
169              var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
170              result = Interval.Divide(result, argumentInterval);
171            }
172
173            break;
174          }
175        case OpCodes.Sin: {
176            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
177            result = Interval.Sine(argumentInterval);
178            break;
179          }
180        case OpCodes.Cos: {
181            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
182            result = Interval.Cosine(argumentInterval);
183            break;
184          }
185        case OpCodes.Tan: {
186            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
187            result = Interval.Tangens(argumentInterval);
188            break;
189          }
190        case OpCodes.Tanh: {
191            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
192            result = Interval.HyperbolicTangent(argumentInterval);
193            break;
194          }
195        case OpCodes.Log: {
196            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
197            result = Interval.Logarithm(argumentInterval);
198            break;
199          }
200        case OpCodes.Exp: {
201            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
202            result = Interval.Exponential(argumentInterval);
203            break;
204          }
205        case OpCodes.Square: {
206            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
207            result = Interval.Square(argumentInterval);
208            break;
209          }
210        case OpCodes.SquareRoot: {
211            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
212            result = Interval.SquareRoot(argumentInterval);
213            break;
214          }
215        case OpCodes.Cube: {
216            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
217            result = Interval.Cube(argumentInterval);
218            break;
219          }
220        case OpCodes.CubeRoot: {
221            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
222            result = Interval.CubicRoot(argumentInterval);
223            break;
224          }
225        case OpCodes.Absolute: {
226            var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
227            result = Interval.Absolute(argumentInterval);
228            break;
229          }
230        case OpCodes.AnalyticQuotient: {
231            result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
232            for (var i = 1; i < currentInstr.nArguments; i++) {
233              var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals);
234              result = Interval.AnalyticQuotient(result, argumentInterval);
235            }
236
237            break;
238          }
239        default:
240          throw new NotSupportedException(
241            $"The tree contains the unknown symbol {currentInstr.dynamicNode.Symbol}");
242      }
243
244      if (!(nodeIntervals == null || nodeIntervals.ContainsKey(currentInstr.dynamicNode)))
245        nodeIntervals.Add(currentInstr.dynamicNode, result);
246
247      return result;
248    }
249
250    #endregion
251
252    #region Helpers
253
254    private static IDictionary<string, Interval> GetOccurringVariableRanges(
255      ISymbolicExpressionTree tree, IntervalCollection variableRanges) {
256      var variables = tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(v => v.VariableName).Distinct()
257                          .ToList();
258
259      return variables.ToDictionary(x => x, x => variableRanges.GetReadonlyDictionary()[x]);
260    }
261
262    #endregion
263 
264    public Interval GetModelBound(ISymbolicExpressionTree tree, IntervalCollection variableRanges) {
265      lock (syncRoot) {
266        EvaluatedSolutions++;
267      }
268
269      var occuringVariableRanges = GetOccurringVariableRanges(tree, variableRanges);
270      var instructions = PrepareInterpreterState(tree, occuringVariableRanges);
271      Interval resultInterval;
272      var instructionCounter = 0;
273      resultInterval = Evaluate(instructions, ref instructionCounter, variableIntervals: occuringVariableRanges);
274
275      // because of numerical errors the bounds might be incorrect
276      if (resultInterval.IsInfiniteOrUndefined || resultInterval.LowerBound <= resultInterval.UpperBound)
277        return resultInterval;
278
279      return new Interval(resultInterval.UpperBound, resultInterval.LowerBound);
280    }
281
282    public IDictionary<ISymbolicExpressionTreeNode, Interval> GetModelNodeBounds(
283      ISymbolicExpressionTree tree, IntervalCollection variableRanges) {
284      throw new NotImplementedException();
285    }
286
287    public double GetConstraintViolation(
288      ISymbolicExpressionTree tree, IntervalCollection variableRanges, ShapeConstraint constraint) {
289      var occuringVariableRanges = GetOccurringVariableRanges(tree, variableRanges);
290      var instructions = PrepareInterpreterState(tree, occuringVariableRanges);
291      var instructionCounter = 0;
292      var modelBound = Evaluate(instructions, ref instructionCounter, variableIntervals: occuringVariableRanges);
293      if (constraint.Interval.Contains(modelBound)) return 0.0;
294
295
296      var error = 0.0;
297
298      if (!constraint.Interval.Contains(modelBound.LowerBound)) {
299        error += Math.Abs(modelBound.LowerBound - constraint.Interval.LowerBound);
300      }
301
302      if (!constraint.Interval.Contains(modelBound.UpperBound)) {
303        error += Math.Abs(modelBound.UpperBound - constraint.Interval.UpperBound);
304      }
305
306      return error;
307    }
308
309
310    public bool IsCompatible(ISymbolicExpressionTree tree) {
311      var containsUnknownSymbols = (
312        from n in tree.Root.GetSubtree(0).IterateNodesPrefix()
313        where
314          !(n.Symbol is Variable) &&
315          !(n.Symbol is Constant) &&
316          !(n.Symbol is StartSymbol) &&
317          !(n.Symbol is Addition) &&
318          !(n.Symbol is Subtraction) &&
319          !(n.Symbol is Multiplication) &&
320          !(n.Symbol is Division) &&
321          !(n.Symbol is Sine) &&
322          !(n.Symbol is Cosine) &&
323          !(n.Symbol is Tangent) &&
324          !(n.Symbol is HyperbolicTangent) &&
325          !(n.Symbol is Logarithm) &&
326          !(n.Symbol is Exponential) &&
327          !(n.Symbol is Square) &&
328          !(n.Symbol is SquareRoot) &&
329          !(n.Symbol is Cube) &&
330          !(n.Symbol is CubeRoot) &&
331          !(n.Symbol is Absolute) &&
332          !(n.Symbol is AnalyticQuotient)
333        select n).Any();
334      return !containsUnknownSymbols;
335    }
336  }
337}
Note: See TracBrowser for help on using the repository browser.