Changeset 17768 for branches/3073_IA_constraint_splitting/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Interpreter
- Timestamp:
- 10/02/20 11:36:34 (4 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
branches/3073_IA_constraint_splitting/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Interpreter/IABoundsEstimator.cs
r17763 r17768 3 3 using System.Collections.ObjectModel; 4 4 using System.Linq; 5 using System.Text;6 using System.Threading.Tasks;7 5 using HEAL.Attic; 8 6 using HeuristicLab.Common; … … 15 13 [StorableType("C8539434-6FB0-47D0-9F5A-2CAE5D8B8B4F")] 16 14 [Item("IA Bounds Estimator", "Interpreter for calculation of intervals of symbolic models.")] 17 public sealed class IABoundsEstimator : ParameterizedNamedItem, IBoundsEstimator {15 public sealed class IABoundsEstimator : ParameterizedNamedItem, IBoundsEstimator { 18 16 #region Parameters 19 17 … … 54 52 set => SplittingWidthParameter.Value.Value = value; 55 53 } 54 56 55 #endregion 57 56 … … 60 59 [StorableConstructor] 61 60 private IABoundsEstimator(StorableConstructorFlag _) : base(_) { } 62 61 63 62 private IABoundsEstimator(IABoundsEstimator original, Cloner cloner) : base(original, cloner) { } 64 63 65 public IABoundsEstimator() : base("IA Bounds Estimator", "Estimates the bounds of the model with interval arithmetic") { 66 Parameters.Add(new FixedValueParameter<IntValue>(EvaluatedSolutionsParameterName, "A counter for the total number of solutions the estimator has evaluated.", new IntValue(0))); 67 Parameters.Add(new FixedValueParameter<BoolValue>(UseIntervalSplittingParameterName, "Defines whether interval splitting is activated or not.", new BoolValue(false))); 68 Parameters.Add(new FixedValueParameter<IntValue>(SplittingIterationsParameterName, "Defines the number of iterations of splitting.", new IntValue(200))); 69 Parameters.Add(new FixedValueParameter<DoubleValue>(SplittingWidthParameterName, "Width of interval, after the splitting should stop.", new DoubleValue(0.0))); 64 public IABoundsEstimator() : base("IA Bounds Estimator", 65 "Estimates the bounds of the model with interval arithmetic") { 66 Parameters.Add(new FixedValueParameter<IntValue>(EvaluatedSolutionsParameterName, 67 "A counter for the total number of solutions the estimator has evaluated.", new IntValue(0))); 68 Parameters.Add(new FixedValueParameter<BoolValue>(UseIntervalSplittingParameterName, 69 "Defines whether interval splitting is activated or not.", new BoolValue(false))); 70 Parameters.Add(new FixedValueParameter<IntValue>(SplittingIterationsParameterName, 71 "Defines the number of iterations of splitting.", new IntValue(200))); 72 Parameters.Add(new FixedValueParameter<DoubleValue>(SplittingWidthParameterName, 73 "Width of interval, after the splitting should stop.", new DoubleValue(0.0))); 70 74 } 71 75 … … 74 78 } 75 79 76 80 #endregion 77 81 78 82 #region IStatefulItem Members … … 86 90 public void ClearState() { } 87 91 88 92 #endregion 89 93 90 94 #region Evaluation … … 261 265 } 262 266 263 267 #endregion 264 268 265 269 #region Helpers 266 270 267 private static IDictionary<string, Interval> GetOccurringVariableRanges(ISymbolicExpressionTree tree, IntervalCollection variableRanges) { 271 private static IDictionary<string, Interval> GetOccurringVariableRanges( 272 ISymbolicExpressionTree tree, IntervalCollection variableRanges) { 268 273 var variables = tree.IterateNodesPrefix().OfType<VariableTreeNode>().Select(v => v.VariableName).Distinct() 269 274 .ToList(); … … 321 326 public static Interval EvaluateWithSplitting(Instruction[] instructions, 322 327 IDictionary<string, Interval> variableIntervals, 323 List<string> multipleOccurenceVariables, int splittingIterations, double splittingWidth, IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null) { 324 var savedIntervals = variableIntervals.ToDictionary(entry => entry.Key, entry => entry.Value); 325 var min = FindBound(instructions, variableIntervals, multipleOccurenceVariables, splittingIterations, splittingWidth, nodeIntervals, 328 List<string> multipleOccurenceVariables, int splittingIterations, 329 double splittingWidth, 330 IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = 331 null) { 332 var min = FindBound(instructions, variableIntervals.ToDictionary(entry => entry.Key, entry => entry.Value), 333 multipleOccurenceVariables, splittingIterations, splittingWidth, nodeIntervals, 326 334 minimization: true); 327 var max = FindBound(instructions, savedIntervals, multipleOccurenceVariables, splittingIterations, splittingWidth, nodeIntervals, 335 var max = FindBound(instructions, variableIntervals.ToDictionary(entry => entry.Key, entry => entry.Value), 336 multipleOccurenceVariables, splittingIterations, splittingWidth, nodeIntervals, 328 337 minimization: false); 329 338 … … 333 342 private static double FindBound(Instruction[] instructions, 334 343 IDictionary<string, Interval> variableIntervals, 335 List<string> multipleOccurenceVariables, int splittingIterations, double splittingWidth, IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null, bool minimization = true) { 344 List<string> multipleOccurenceVariables, int splittingIterations, 345 double splittingWidth, 346 IDictionary<ISymbolicExpressionTreeNode, Interval> nodeIntervals = null, 347 bool minimization = true, bool stopAtLimit = false, double limit = 0) { 336 348 SortedSet<BoxBound> prioQ = new SortedSet<BoxBound>(); 337 338 349 var ic = 0; 350 var stop = false; 339 351 //Calculate full box 340 //IReadOnlyDictionary<string, Interval> readonlyRanges = 341 // variableIntervals.ToDictionary(k => k.Key, k => k.Value); 342 var interval = Evaluate(instructions, ref ic, nodeIntervals, variableIntervals:variableIntervals); 352 var interval = Evaluate(instructions, ref ic, nodeIntervals, variableIntervals: variableIntervals); 343 353 // the order of keys in a dictionary is guaranteed to be the same order as values in a dictionary 344 354 // https://docs.microsoft.com/en-us/dotnet/api/system.collections.idictionary.keys?view=netcore-3.1#remarks … … 348 358 if (minimization) { 349 359 prioQ.Add(new BoxBound(box, interval.LowerBound)); 360 if (stopAtLimit && interval.LowerBound >= limit) stop = true; 350 361 } else { 351 362 prioQ.Add(new BoxBound(box, -interval.UpperBound)); 363 if (stopAtLimit && interval.UpperBound <= limit) stop = true; 352 364 } 353 365 354 366 var discardedBound = double.MaxValue; 355 367 var runningBound = double.MaxValue; 356 for (var depth = 0; depth < splittingIterations && prioQ.Count > 0 ; ++depth) {368 for (var depth = 0; depth < splittingIterations && prioQ.Count > 0 && !stop; ++depth) { 357 369 var currentBound = prioQ.Min; 358 370 prioQ.Remove(currentBound); … … 382 394 var res = Evaluate(instructions, ref ic, nodeIntervals, 383 395 new ReadOnlyDictionary<string, Interval>(variableIntervals)); 384 //if (minimization) { 385 // prioQ.Add(new BoxBound(newBox, res.LowerBound)); 386 //} else { 387 // prioQ.Add(new BoxBound(newBox, -res.UpperBound)); 388 //} 396 389 397 var boxBound = new BoxBound(newBox, minimization ? res.LowerBound : -res.UpperBound); 390 398 prioQ.Add(boxBound); … … 394 402 runningBound = innerBound; 395 403 404 if (minimization) { 405 if (stopAtLimit && innerBound >= limit) 406 stop = true; 407 } else { 408 if (stopAtLimit && innerBound <= limit) 409 stop = true; 410 } 396 411 } 397 412 … … 413 428 414 429 private static IEnumerable<IEnumerable<Interval>> Split(List<Interval> box, double minWidth) { 415 List<Interval> toList(Tuple<Interval, Interval> t) => new List<Interval> {t.Item1, t.Item2};430 List<Interval> toList(Tuple<Interval, Interval> t) => new List<Interval> {t.Item1, t.Item2}; 416 431 var boxes = box.Select(region => region.Width > minWidth ? toList(region.Split()) : new List<Interval> {region}) 417 432 .ToList(); … … 435 450 } else { 436 451 var vars = ContainsVariableMultipleTimes(tree, out var variables); 437 resultInterval = EvaluateWithSplitting(instructions, occuringVariableRanges, variables, SplittingIterations, SplittingWidth); 452 resultInterval = EvaluateWithSplitting(instructions, occuringVariableRanges, variables, SplittingIterations, 453 SplittingWidth); 438 454 } 439 455 … … 445 461 } 446 462 447 public IDictionary<ISymbolicExpressionTreeNode, Interval> GetModelNodesBounds(ISymbolicExpressionTree tree, IntervalCollection variableRanges) { 463 public IDictionary<ISymbolicExpressionTreeNode, Interval> GetModelNodesBounds( 464 ISymbolicExpressionTree tree, IntervalCollection variableRanges) { 448 465 throw new NotImplementedException(); 449 466 } 450 467 451 468 public double CheckConstraint( 469 ISymbolicExpressionTree tree, IntervalCollection variableRanges, IntervalConstraint constraint) { 470 var occuringVariableRanges = GetOccurringVariableRanges(tree, variableRanges); 471 var instructions = PrepareInterpreterState(tree, occuringVariableRanges); 472 if (!UseIntervalSplitting) { 473 var instructionCounter = 0; 474 var modelBound = Evaluate(instructions, ref instructionCounter, variableIntervals: occuringVariableRanges); 475 if (constraint.Interval.Contains(modelBound)) return 0.0; 476 return Math.Abs(modelBound.LowerBound - constraint.Interval.LowerBound) + 477 Math.Abs(modelBound.UpperBound - constraint.Interval.UpperBound); 478 } 479 480 if (double.IsNegativeInfinity(constraint.Interval.LowerBound) && 481 double.IsPositiveInfinity(constraint.Interval.UpperBound)) { 482 return 0.0; 483 } 484 485 ContainsVariableMultipleTimes(tree, out var variables); 486 487 var upperBound = 0.0; 488 var lowerBound = 0.0; 489 if (double.IsNegativeInfinity(constraint.Interval.LowerBound)) { 490 upperBound = FindBound(instructions, occuringVariableRanges, variables, SplittingIterations, SplittingWidth, 491 minimization: false, stopAtLimit: true, limit: constraint.Interval.UpperBound); 492 493 return upperBound <= constraint.Interval.UpperBound 494 ? 0.0 495 : Math.Abs(upperBound - constraint.Interval.UpperBound); 496 } 497 498 if (double.IsPositiveInfinity(constraint.Interval.UpperBound)) { 499 lowerBound = FindBound(instructions, occuringVariableRanges, variables, SplittingIterations, SplittingWidth, 500 minimization: true, stopAtLimit: true, limit: constraint.Interval.LowerBound); 501 502 return lowerBound <= constraint.Interval.LowerBound 503 ? 0.0 504 : Math.Abs(lowerBound - constraint.Interval.LowerBound); 505 } 506 507 upperBound = FindBound(instructions, occuringVariableRanges, variables, SplittingIterations, SplittingWidth, 508 minimization: false, stopAtLimit: true, limit: constraint.Interval.UpperBound); 509 lowerBound = FindBound(instructions, occuringVariableRanges, variables, SplittingIterations, SplittingWidth, 510 minimization: true, stopAtLimit: true, limit: constraint.Interval.LowerBound); 511 512 513 var res = 0.0; 514 515 res += upperBound <= constraint.Interval.UpperBound ? 0.0 : Math.Abs(upperBound - constraint.Interval.UpperBound); 516 res += lowerBound <= constraint.Interval.LowerBound ? 0.0 : Math.Abs(lowerBound - constraint.Interval.LowerBound); 517 518 return res; 519 } 520 521 522 public bool IsCompatible(ISymbolicExpressionTree tree) { 523 var containsUnknownSymbols = ( 524 from n in tree.Root.GetSubtree(0).IterateNodesPrefix() 525 where 526 !(n.Symbol is Variable) && 527 !(n.Symbol is Constant) && 528 !(n.Symbol is StartSymbol) && 529 !(n.Symbol is Addition) && 530 !(n.Symbol is Subtraction) && 531 !(n.Symbol is Multiplication) && 532 !(n.Symbol is Division) && 533 !(n.Symbol is Sine) && 534 !(n.Symbol is Cosine) && 535 !(n.Symbol is Tangent) && 536 !(n.Symbol is HyperbolicTangent) && 537 !(n.Symbol is Logarithm) && 538 !(n.Symbol is Exponential) && 539 !(n.Symbol is Square) && 540 !(n.Symbol is SquareRoot) && 541 !(n.Symbol is Cube) && 542 !(n.Symbol is CubeRoot) && 543 !(n.Symbol is Absolute) && 544 !(n.Symbol is AnalyticQuotient) 545 select n).Any(); 546 return !containsUnknownSymbols; 547 } 452 548 } 453 549 }
Note: See TracChangeset
for help on using the changeset viewer.