Changeset 17631 for branches/3073_IA_constraint_splitting/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Interpreter
- Timestamp:
- 06/24/20 16:12:03 (5 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
branches/3073_IA_constraint_splitting/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Interpreter/IntervalInterpreter.cs
r17628 r17631 1 1 #region License Information 2 2 3 /* HeuristicLab 3 4 * Copyright (C) Heuristic and Evolutionary Algorithms Laboratory (HEAL) … … 18 19 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>. 19 20 */ 21 20 22 #endregion 21 23 … … 23 25 using System.Collections.Generic; 24 26 using System.Linq; 27 using System.Runtime.Remoting.Contexts; 25 28 using HEAL.Attic; 26 29 using HeuristicLab.Common; … … 32 35 namespace HeuristicLab.Problems.DataAnalysis.Symbolic { 33 36 [StorableType("DE6C1E1E-D7C1-4070-847E-63B68562B10C")] 34 [Item("IntervalInterpreter", "Int perter for calculation of intervals of symbolic models.")]37 [Item("IntervalInterpreter", "Interpreter for calculation of intervals of symbolic models.")] 35 38 public sealed class IntervalInterpreter : ParameterizedNamedItem, IStatefulItem { 36 37 39 private const string EvaluatedSolutionsParameterName = "EvaluatedSolutions"; 38 39 public IFixedValueParameter<IntValue> EvaluatedSolutionsParameter => (IFixedValueParameter<IntValue>)Parameters[EvaluatedSolutionsParameterName]; 40 private const string MinSplittingWidthParameterName = "MinSplittingWidth"; 41 private const string MaxSplittingDepthParameterName = "MaxSplittingDepth"; 42 private const string UseIntervalSplittingParameterName = "UseIntervalSplitting"; 43 44 public IFixedValueParameter<IntValue> EvaluatedSolutionsParameter => 45 (IFixedValueParameter<IntValue>) Parameters[EvaluatedSolutionsParameterName]; 46 47 public IFixedValueParameter<IntValue> MinSplittingWithParameter => 48 (IFixedValueParameter<IntValue>) Parameters[MinSplittingWidthParameterName]; 49 50 public IFixedValueParameter<IntValue> MaxSplittingDepthParameter => 51 (IFixedValueParameter<IntValue>) Parameters[MaxSplittingDepthParameterName]; 52 53 public IFixedValueParameter<BoolValue> UseIntervalSplittingParameter => 54 (IFixedValueParameter<BoolValue>) Parameters[UseIntervalSplittingParameterName]; 55 56 public int MinSplittingWidth { 57 get => MinSplittingWithParameter.Value.Value; 58 set => MinSplittingWithParameter.Value.Value = value; 59 } 60 61 public int MaxSplittingDepth { 62 get => MaxSplittingDepthParameter.Value.Value; 63 set => MaxSplittingDepthParameter.Value.Value = value; 64 } 65 66 public bool UseIntervalSplitting { 67 get => UseIntervalSplittingParameter.Value.Value; 68 set => UseIntervalSplittingParameter.Value.Value = value; 69 } 40 70 41 71 public int EvaluatedSolutions { … … 46 76 [StorableConstructor] 47 77 private IntervalInterpreter(StorableConstructorFlag _) : base(_) { } 78 48 79 private IntervalInterpreter(IntervalInterpreter original, Cloner cloner) 49 80 : base(original, cloner) { } 50 81 51 82 public IntervalInterpreter() 52 : base("IntervalInterpreter", "Intperter for calculation of intervals of symbolic models.") { 53 Parameters.Add(new FixedValueParameter<IntValue>(EvaluatedSolutionsParameterName, "A counter for the total number of solutions the interpreter has evaluated", new IntValue(0))); 83 : base("IntervalInterpreter", "Interpreter for calculation of intervals of symbolic models.") { 84 Parameters.Add(new FixedValueParameter<IntValue>(EvaluatedSolutionsParameterName, 85 "A counter for the total number of solutions the interpreter has evaluated", new IntValue(0))); 86 Parameters.Add(new FixedValueParameter<IntValue>(MinSplittingWidthParameterName, "Minimum interval width until splitting is stopped", new IntValue(0))); 87 Parameters.Add(new FixedValueParameter<IntValue>(MaxSplittingDepthParameterName, "Maximum recursion depth of the splitting", new IntValue(5))); 88 Parameters.Add(new FixedValueParameter<BoolValue>(UseIntervalSplittingParameterName, "", new BoolValue(false))); 54 89 } 55 90 … … 61 96 62 97 #region IStatefulItem Members 98 63 99 public void InitializeState() { 64 100 EvaluatedSolutions = 0; 65 101 } 102 66 103 public void ClearState() { } 104 67 105 #endregion 68 106 69 public Interval GetSymbolicExpressionTreeInterval(ISymbolicExpressionTree tree, IDataset dataset, IEnumerable<int> rows = null) { 107 public Interval GetSymbolicExpressionTreeInterval( 108 ISymbolicExpressionTree tree, IDataset dataset, 109 IEnumerable<int> rows = null, bool splitting = true) { 70 110 var variableRanges = DatasetUtil.GetVariableRanges(dataset, rows); 71 return GetSymbolicExpressionTreeInterval(tree, variableRanges); 72 } 73 74 public Interval GetSymbolicExpressionTreeIntervals(ISymbolicExpressionTree tree, IDataset dataset, 75 out IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals, IEnumerable<int> rows = null) { 111 return GetSymbolicExpressionTreeInterval(tree, variableRanges, splitting); 112 } 113 114 public Interval GetSymbolicExpressionTreeIntervals( 115 ISymbolicExpressionTree tree, IDataset dataset, 116 out IDictionary<ISymbolicExpressionTreeNode, Interval> 117 nodeIntervals, IEnumerable<int> rows = null, bool splitting = true) { 76 118 var variableRanges = DatasetUtil.GetVariableRanges(dataset, rows); 77 return GetSymbolicExpressionTreeIntervals(tree, variableRanges, out nodeIntervals); 78 } 79 80 public Interval GetSymbolicExpressionTreeInterval(ISymbolicExpressionTree tree, IReadOnlyDictionary<string, Interval> variableRanges) { 119 return GetSymbolicExpressionTreeIntervals(tree, variableRanges, out nodeIntervals, splitting); 120 } 121 122 public Interval GetSymbolicExpressionTreeInterval( 123 ISymbolicExpressionTree tree, 124 IDictionary<string, Interval> variableRanges, bool splitting = true) { 81 125 lock (syncRoot) { 82 126 EvaluatedSolutions++; 83 127 } 84 int instructionCount = 0; 85 var instructions = PrepareInterpreterState(tree, variableRanges); 86 var outputInterval = Evaluate(instructions, ref instructionCount); 87 88 // because of numerical errors the bounds might be incorrect 89 return outputInterval.LowerBound <= outputInterval.UpperBound ? outputInterval : new Interval(outputInterval.UpperBound, outputInterval.LowerBound); 90 } 91 92 93 public Interval GetSymbolicExpressionTreeIntervals(ISymbolicExpressionTree tree, 94 IReadOnlyDictionary<string, Interval> variableRanges, out IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals) { 128 129 Interval outputInterval; 130 131 if (splitting) { 132 outputInterval = GetSymbolicExpressionTreeIntervals(tree, variableRanges, 133 out var nodeIntervals); 134 } else { 135 var instructionCount = 0; 136 var instructions = PrepareInterpreterState(tree, variableRanges); 137 outputInterval = Evaluate(instructions, ref instructionCount); 138 } 139 140 return outputInterval.LowerBound <= outputInterval.UpperBound 141 ? outputInterval 142 : new Interval(outputInterval.UpperBound, outputInterval.LowerBound); 143 } 144 145 146 public Interval GetSymbolicExpressionTreeIntervals( 147 ISymbolicExpressionTree tree, 148 IDictionary<string, Interval> variableRanges, 149 out IDictionary<ISymbolicExpressionTreeNode, Interval> 150 nodeIntervals, bool splitting = true) { 95 151 lock (syncRoot) { 96 152 EvaluatedSolutions++; 97 153 } 98 int instructionCount = 0; 154 99 155 var intervals = new Dictionary<ISymbolicExpressionTreeNode, Interval>(); 100 156 var instructions = PrepareInterpreterState(tree, variableRanges); 101 var outputInterval = Evaluate(instructions, ref instructionCount, intervals); 102 103 // fix incorrect intervals if necessary (could occur because of numerical errors) 157 158 Interval outputInterval; 159 if (splitting) { 160 var variables = tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(x => x.VariableName).Distinct() 161 .ToList(); 162 var containsDependencyProblem = ContainsVariableMultipleTimes(tree); 163 164 if (variables.Count > 1 && containsDependencyProblem) { 165 var currIndex = 0; 166 var currDepth = 0; 167 outputInterval = EvaluateRecursive(instructions, intervals, variableRanges, variables, MinSplittingWidth, MaxSplittingDepth, 168 ref currIndex, ref currDepth, tree); 169 } else { 170 var instructionCount = 0; 171 outputInterval = Evaluate(instructions, ref instructionCount, intervals); 172 } 173 } else { 174 var instructionCount = 0; 175 outputInterval = Evaluate(instructions, ref instructionCount, intervals); 176 } 177 104 178 nodeIntervals = new Dictionary<ISymbolicExpressionTreeNode, Interval>(); 105 179 foreach (var kvp in intervals) { … … 114 188 if (outputInterval.IsInfiniteOrUndefined || outputInterval.LowerBound <= outputInterval.UpperBound) 115 189 return outputInterval; 116 else 117 return new Interval(outputInterval.UpperBound, outputInterval.LowerBound); 118 } 119 120 121 private static Instruction[] PrepareInterpreterState(ISymbolicExpressionTree tree, IReadOnlyDictionary<string, Interval> variableRanges) { 190 191 return new Interval(outputInterval.UpperBound, outputInterval.LowerBound); 192 } 193 194 195 private static Instruction[] PrepareInterpreterState( 196 ISymbolicExpressionTree tree, 197 IDictionary<string, Interval> variableRanges) { 122 198 if (variableRanges == null) 123 199 throw new ArgumentNullException("No variablew ranges are present!", nameof(variableRanges)); 124 200 125 201 //Check if all variables used in the tree are present in the dataset 126 foreach (var variable in tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(n => n.VariableName).Distinct()) {127 if (!variableRanges.ContainsKey(variable)) throw new InvalidOperationException($"No ranges for variable {variable} is present");128 }129 130 Instruction[]code = SymbolicExpressionTreeCompiler.Compile(tree, OpCodes.MapSymbolToOpCode);131 foreach ( Instructioninstr in code.Where(i => i.opCode == OpCodes.Variable)) {132 var variableTreeNode = (VariableTreeNode) instr.dynamicNode;202 foreach (var variable in tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(n => n.VariableName).Distinct()) 203 if (!variableRanges.ContainsKey(variable)) 204 throw new InvalidOperationException($"No ranges for variable {variable} is present"); 205 206 var code = SymbolicExpressionTreeCompiler.Compile(tree, OpCodes.MapSymbolToOpCode); 207 foreach (var instr in code.Where(i => i.opCode == OpCodes.Variable)) { 208 var variableTreeNode = (VariableTreeNode) instr.dynamicNode; 133 209 instr.data = variableRanges[variableTreeNode.VariableName]; 134 210 } 211 135 212 return code; 136 213 } 137 214 138 private Interval Evaluate(Instruction[] instructions, ref int instructionCounter, IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null) { 139 Instruction currentInstr = instructions[instructionCounter]; 215 public static Interval EvaluateRecursive( 216 Instruction[] instructions, 217 IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals, 218 IDictionary<string, Interval> variableIntervals, IList<string> variables, 219 double minWidth, int maxDepth, ref int currIndex, ref int currDepth, 220 ISymbolicExpressionTree tree) { 221 Interval evaluate() { 222 var ic = 0; 223 return Evaluate(instructions, ref ic, nodeIntervals, variableIntervals); 224 } 225 226 Interval recurse(ref int idx, ref int depth) { 227 return EvaluateRecursive(instructions, nodeIntervals, variableIntervals, variables, minWidth, maxDepth, ref idx, 228 ref depth, tree); 229 } 230 231 232 var v = variables[currIndex]; 233 var x = variableIntervals[v]; 234 if (x.Width < minWidth || currDepth == maxDepth || !MultipleTimes(tree, v)) { 235 if (currIndex + 1 < variables.Count) { 236 currDepth = 0; 237 currIndex++; 238 var z = recurse(ref currIndex, ref currDepth); 239 currIndex--; 240 return z; 241 } 242 243 return evaluate(); 244 } 245 246 var t = x.Split(); 247 var xa = t.Item1; 248 var xb = t.Item2; 249 var d = currDepth; 250 currDepth = d + 1; 251 variableIntervals[v] = xa; 252 var ya = recurse(ref currIndex, ref currDepth); 253 currDepth = d + 1; 254 variableIntervals[v] = xb; 255 var yb = recurse(ref currIndex, ref currDepth); 256 variableIntervals[v] = x; // restore interval 257 return ya | yb; 258 } 259 260 public static Interval Evaluate( 261 Instruction[] instructions, ref int instructionCounter, 262 IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null, 263 IDictionary<string, Interval> variableIntervals = null) { 264 var currentInstr = instructions[instructionCounter]; 140 265 //Use ref parameter, because the tree will be iterated through recursively from the left-side branch to the right side 141 266 //Update instructionCounter, whenever Evaluate is called … … 146 271 //Variables, Constants, ... 147 272 case OpCodes.Variable: { 148 var variableTreeNode = (VariableTreeNode)currentInstr.dynamicNode; 149 var weightInterval = new Interval(variableTreeNode.Weight, variableTreeNode.Weight); 150 var variableInterval = (Interval)currentInstr.data; 151 152 result = Interval.Multiply(variableInterval, weightInterval); 153 break; 154 } 273 var variableTreeNode = (VariableTreeNode) currentInstr.dynamicNode; 274 var weightInterval = new Interval(variableTreeNode.Weight, variableTreeNode.Weight); 275 //var variableInterval = (Interval)currentInstr.data; 276 277 Interval variableInterval; 278 if (variableIntervals != null && variableIntervals.ContainsKey(variableTreeNode.VariableName)) 279 variableInterval = variableIntervals[variableTreeNode.VariableName]; 280 else 281 variableInterval = (Interval) currentInstr.data; 282 283 result = Interval.Multiply(variableInterval, weightInterval); 284 break; 285 } 155 286 case OpCodes.Constant: { 156 var constTreeNode = (ConstantTreeNode)currentInstr.dynamicNode;157 158 159 287 var constTreeNode = (ConstantTreeNode) currentInstr.dynamicNode; 288 result = new Interval(constTreeNode.Value, constTreeNode.Value); 289 break; 290 } 160 291 //Elementary arithmetic rules 161 292 case OpCodes.Add: { 162 163 for (int i = 1; i < currentInstr.nArguments; i++) {164 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals);165 result = Interval.Add(result, argumentInterval);166 }167 break;293 //result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 294 result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 295 for (var i = 1; i < currentInstr.nArguments; i++) { 296 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 297 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 298 result = Interval.Add(result, argumentInterval); 168 299 } 300 301 break; 302 } 169 303 case OpCodes.Sub: { 170 171 if (currentInstr.nArguments == 1)172 result = Interval.Multiply(new Interval(-1, -1), result);173 174 for (int i = 1; i < currentInstr.nArguments; i++) { 175 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals);176 result = Interval.Subtract(result, argumentInterval);177 }178 break;304 //result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 305 result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 306 if (currentInstr.nArguments == 1) 307 result = Interval.Multiply(new Interval(-1, -1), result); 308 309 for (var i = 1; i < currentInstr.nArguments; i++) { 310 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 311 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 312 result = Interval.Subtract(result, argumentInterval); 179 313 } 314 315 break; 316 } 180 317 case OpCodes.Mul: { 181 182 for (int i = 1; i < currentInstr.nArguments; i++) {183 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals);184 result = Interval.Multiply(result, argumentInterval);185 }186 break;318 //result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 319 result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 320 for (var i = 1; i < currentInstr.nArguments; i++) { 321 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 322 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 323 result = Interval.Multiply(result, argumentInterval); 187 324 } 325 326 break; 327 } 188 328 case OpCodes.Div: { 189 190 if (currentInstr.nArguments == 1)191 result = Interval.Divide(new Interval(1, 1), result);192 193 for (int i = 1; i < currentInstr.nArguments; i++) { 194 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals);195 result = Interval.Divide(result, argumentInterval);196 }197 break;329 //result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 330 result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 331 if (currentInstr.nArguments == 1) 332 result = Interval.Divide(new Interval(1, 1), result); 333 334 for (var i = 1; i < currentInstr.nArguments; i++) { 335 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 336 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 337 result = Interval.Divide(result, argumentInterval); 198 338 } 339 340 break; 341 } 199 342 //Trigonometric functions 200 343 case OpCodes.Sin: { 201 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 202 result = Interval.Sine(argumentInterval); 203 break; 204 } 344 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 345 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 346 result = Interval.Sine(argumentInterval); 347 break; 348 } 205 349 case OpCodes.Cos: { 206 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 207 result = Interval.Cosine(argumentInterval); 208 break; 209 } 350 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 351 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 352 result = Interval.Cosine(argumentInterval); 353 break; 354 } 210 355 case OpCodes.Tan: { 211 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 212 result = Interval.Tangens(argumentInterval); 213 break; 214 } 356 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 357 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 358 result = Interval.Tangens(argumentInterval); 359 break; 360 } 215 361 case OpCodes.Tanh: { 216 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 217 result = Interval.HyperbolicTangent(argumentInterval); 218 break; 219 } 362 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 363 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 364 result = Interval.HyperbolicTangent(argumentInterval); 365 break; 366 } 220 367 //Exponential functions 221 368 case OpCodes.Log: { 222 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 223 result = Interval.Logarithm(argumentInterval); 224 break; 369 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 370 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 371 result = Interval.Logarithm(argumentInterval); 372 break; 373 } 374 case OpCodes.Exp: { 375 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 376 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 377 result = Interval.Exponential(argumentInterval); 378 break; 379 } 380 case OpCodes.Square: { 381 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 382 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 383 result = Interval.Square(argumentInterval); 384 break; 385 } 386 case OpCodes.SquareRoot: { 387 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 388 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 389 result = Interval.SquareRoot(argumentInterval); 390 break; 391 } 392 case OpCodes.Cube: { 393 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 394 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 395 result = Interval.Cube(argumentInterval); 396 break; 397 } 398 case OpCodes.CubeRoot: { 399 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 400 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 401 result = Interval.CubicRoot(argumentInterval); 402 break; 403 } 404 case OpCodes.Absolute: { 405 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 406 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 407 result = Interval.Absolute(argumentInterval); 408 break; 409 } 410 case OpCodes.AnalyticQuotient: { 411 //result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 412 result = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 413 for (var i = 1; i < currentInstr.nArguments; i++) { 414 //var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 415 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals, variableIntervals); 416 result = Interval.AnalyticalQuotient(result, argumentInterval); 225 417 } 226 case OpCodes.Exp: { 227 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 228 result = Interval.Exponential(argumentInterval); 229 break; 230 } 231 case OpCodes.Square: { 232 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 233 result = Interval.Square(argumentInterval); 234 break; 235 } 236 case OpCodes.SquareRoot: { 237 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 238 result = Interval.SquareRoot(argumentInterval); 239 break; 240 } 241 case OpCodes.Cube: { 242 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 243 result = Interval.Cube(argumentInterval); 244 break; 245 } 246 case OpCodes.CubeRoot: { 247 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 248 result = Interval.CubicRoot(argumentInterval); 249 break; 250 } 251 case OpCodes.Absolute: { 252 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 253 result = Interval.Absolute(argumentInterval); 254 break; 255 } 256 case OpCodes.AnalyticQuotient: { 257 result = Evaluate(instructions, ref instructionCounter, nodeIntervals); 258 for (var i = 1; i < currentInstr.nArguments; i++) { 259 var argumentInterval = Evaluate(instructions, ref instructionCounter, nodeIntervals); 260 result = Interval.AnalyticalQuotient(result, argumentInterval); 261 } 262 263 break; 264 } 418 419 break; 420 } 265 421 default: 266 422 throw new NotSupportedException($"The tree contains the unknown symbol {currentInstr.dynamicNode.Symbol}"); 267 423 } 268 424 269 if ( nodeIntervals != null)425 if (!(nodeIntervals == null || nodeIntervals.ContainsKey(currentInstr.dynamicNode))) 270 426 nodeIntervals.Add(currentInstr.dynamicNode, result); 271 427 272 428 return result; 273 429 } 430 431 private static bool MultipleTimes(ISymbolicExpressionTree tree, string variable) { 432 var varlist = tree.IterateNodesPrefix().OfType<VariableTreeNode>().GroupBy(x => x.VariableName); 433 var group = varlist.Select(x => x.Key == variable).Count(); 434 435 return group > 1; 436 } 437 438 private static bool ContainsVariableMultipleTimes(ISymbolicExpressionTree tree) { 439 var varlist = tree.IterateNodesPrefix().OfType<VariableTreeNode>().GroupBy(x => x.VariableName); 440 return varlist.Any(@group => @group.Count() > 1); 441 } 442 274 443 275 444 public static bool IsCompatible(ISymbolicExpressionTree tree) { … … 277 446 from n in tree.Root.GetSubtree(0).IterateNodesPrefix() 278 447 where 279 !(n.Symbol is Problems.DataAnalysis.Symbolic.Variable) &&448 !(n.Symbol is Variable) && 280 449 !(n.Symbol is Constant) && 281 450 !(n.Symbol is StartSymbol) &&
Note: See TracChangeset
for help on using the changeset viewer.