Free cookie consent management tool by TermsFeed Policy Generator

source: stable/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Formatters/InfixExpressionFormatter.cs @ 17078

Last change on this file since 17078 was 17075, checked in by mkommend, 6 years ago

#2998: Merged r16700 into stable.

File size: 8.6 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2018 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.Globalization;
25using System.Linq;
26using System.Text;
27using HeuristicLab.Common;
28using HeuristicLab.Core;
29using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
30using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
31
32namespace HeuristicLab.Problems.DataAnalysis.Symbolic {
33  /// <summary>
34  /// Formats mathematical expressions in infix form. E.g. x1 * (3.0 * x2 + x3)
35  /// </summary>
36  [StorableClass]
37  [Item("Infix Symbolic Expression Tree Formatter", "A string formatter that converts symbolic expression trees to infix expressions.")]
38
39  public sealed class InfixExpressionFormatter : NamedItem, ISymbolicExpressionTreeStringFormatter {
40
41
42    [StorableConstructor]
43    private InfixExpressionFormatter(bool deserializing) : base(deserializing) { }
44    private InfixExpressionFormatter(InfixExpressionFormatter original, Cloner cloner) : base(original, cloner) { }
45    public InfixExpressionFormatter()
46      : base() {
47      Name = ItemName;
48      Description = ItemDescription;
49    }
50    public override IDeepCloneable Clone(Cloner cloner) {
51      return new InfixExpressionFormatter(this, cloner);
52    }
53
54    /// <summary>
55    /// Produces an infix expression for a given expression tree.
56    /// </summary>
57    /// <param name="symbolicExpressionTree">The tree representation of the expression.</param>
58    /// <param name="numberFormat">Number format that should be used for numeric parameters (e.g. NumberFormatInfo.InvariantInfo (default)).</param>
59    /// <param name="formatString">The format string for numeric parameters (e.g. \"G4\" to limit to 4 digits, default is \"G\")</param>
60    /// <returns>Infix expression</returns>
61    public string Format(ISymbolicExpressionTree symbolicExpressionTree, NumberFormatInfo numberFormat, string formatString="G") {
62      // skip root and start symbols
63      StringBuilder strBuilder = new StringBuilder();
64      FormatRecursively(symbolicExpressionTree.Root.GetSubtree(0).GetSubtree(0), strBuilder, numberFormat, formatString);
65      return strBuilder.ToString();
66    }
67
68    public string Format(ISymbolicExpressionTree symbolicExpressionTree) {
69      return Format(symbolicExpressionTree, NumberFormatInfo.InvariantInfo);
70    }
71
72    private static void FormatRecursively(ISymbolicExpressionTreeNode node, StringBuilder strBuilder, NumberFormatInfo numberFormat, string formatString) {
73      if (node.SubtreeCount > 1) {
74        var token = GetToken(node.Symbol);
75        // operators
76        if (token == "+" || token == "-" || token == "OR" || token == "XOR" ||
77            token == "*" || token == "/" || token == "AND" ||
78            token == "^") {
79          strBuilder.Append("(");
80          FormatRecursively(node.Subtrees.First(), strBuilder, numberFormat, formatString);
81
82          foreach (var subtree in node.Subtrees.Skip(1)) {
83            strBuilder.Append(" ").Append(token).Append(" ");
84            FormatRecursively(subtree, strBuilder, numberFormat, formatString);
85          }
86          strBuilder.Append(")");
87        } else {
88          // function with multiple arguments
89          strBuilder.Append(token).Append("(");
90          FormatRecursively(node.Subtrees.First(), strBuilder, numberFormat, formatString);
91          foreach (var subtree in node.Subtrees.Skip(1)) {
92            strBuilder.Append(", ");
93            FormatRecursively(subtree, strBuilder, numberFormat, formatString);
94          }
95          strBuilder.Append(")");
96        }
97      } else if (node.SubtreeCount == 1) {
98        var token = GetToken(node.Symbol);
99        if (token == "-" || token == "NOT") {
100          strBuilder.Append("(").Append(token).Append("(");
101          FormatRecursively(node.GetSubtree(0), strBuilder, numberFormat, formatString);
102          strBuilder.Append("))");
103        } else if (token == "/") {
104          strBuilder.Append("1/");
105          FormatRecursively(node.GetSubtree(0), strBuilder, numberFormat, formatString);
106        } else if (token == "+" || token == "*") {
107          FormatRecursively(node.GetSubtree(0), strBuilder, numberFormat, formatString);
108        } else {
109          // function with only one argument
110          strBuilder.Append(token).Append("(");
111          FormatRecursively(node.GetSubtree(0), strBuilder, numberFormat, formatString);
112          strBuilder.Append(")");
113        }
114      } else {
115        // no subtrees
116        if (node.Symbol is LaggedVariable) {
117          var varNode = node as LaggedVariableTreeNode;
118          if (!varNode.Weight.IsAlmost(1.0)) {
119            strBuilder.Append("(");
120            strBuilder.Append(varNode.Weight.ToString(formatString, numberFormat));
121            strBuilder.Append("*");
122          }
123          strBuilder.Append("LAG(");
124          if (varNode.VariableName.Contains("'")) {
125            strBuilder.AppendFormat("\"{0}\"", varNode.VariableName);
126          } else {
127            strBuilder.AppendFormat("'{0}'", varNode.VariableName);
128          }
129          strBuilder.Append(", ")
130            .AppendFormat(numberFormat, "{0}", varNode.Lag)
131            .Append(")");
132        } else if (node.Symbol is Variable) {
133          var varNode = node as VariableTreeNode;
134          if (!varNode.Weight.IsAlmost(1.0)) {
135            strBuilder.Append("(");
136            strBuilder.Append(varNode.Weight.ToString(formatString, numberFormat));
137            strBuilder.Append("*");
138          }
139          if (varNode.VariableName.Contains("'")) {
140            strBuilder.AppendFormat("\"{0}\"", varNode.VariableName);
141          } else {
142            strBuilder.AppendFormat("'{0}'", varNode.VariableName);
143          }
144          if (!varNode.Weight.IsAlmost(1.0)) {
145            strBuilder.Append(")");
146          }
147        } else if (node.Symbol is FactorVariable) {
148          var factorNode = node as FactorVariableTreeNode;
149          if (factorNode.VariableName.Contains("'")) {
150            strBuilder.AppendFormat("\"{0}\"", factorNode.VariableName);
151          } else {
152            strBuilder.AppendFormat("'{0}'", factorNode.VariableName);
153          }
154          strBuilder.AppendFormat("[{0}]",
155            string.Join(", ", factorNode.Weights.Select(w => w.ToString(formatString, numberFormat))));
156        } else if (node.Symbol is BinaryFactorVariable) {
157          var factorNode = node as BinaryFactorVariableTreeNode;
158          if (!factorNode.Weight.IsAlmost(1.0)) {
159            strBuilder.Append("(");
160            strBuilder.Append(factorNode.Weight.ToString(formatString, numberFormat));
161            strBuilder.Append("*");
162          }
163          if (factorNode.VariableName.Contains("'")) {
164            strBuilder.AppendFormat("\"{0}\"", factorNode.VariableName);
165          } else {
166            strBuilder.AppendFormat("'{0}'", factorNode.VariableName);
167          }
168          strBuilder.Append(" = ");
169          if (factorNode.VariableValue.Contains("'")) {
170            strBuilder.AppendFormat("\"{0}\"", factorNode.VariableValue);
171          } else {
172            strBuilder.AppendFormat("'{0}'", factorNode.VariableValue);
173          }
174
175          if (!factorNode.Weight.IsAlmost(1.0)) {
176            strBuilder.Append(")");
177          }
178
179        } else if (node.Symbol is Constant) {
180          var constNode = node as ConstantTreeNode;
181          if (constNode.Value >= 0.0)
182            strBuilder.Append(constNode.Value.ToString(formatString, numberFormat));
183          else
184            strBuilder.Append("(").Append(constNode.Value.ToString(formatString, numberFormat)).Append(")"); // (-1
185        }
186      }
187    }
188
189    private static string GetToken(ISymbol symbol) {
190      var tok = InfixExpressionParser.knownSymbols.GetBySecond(symbol).FirstOrDefault();
191      if (tok == null)
192        throw new ArgumentException(string.Format("Unknown symbol {0} found.", symbol.Name));
193      return tok;
194    }
195  }
196}
Note: See TracBrowser for help on using the repository browser.