Free cookie consent management tool by TermsFeed Policy Generator

source: branches/3040_VectorBasedGP/HeuristicLab.Problems.DataAnalysis.Symbolic/3.4/Formatters/InfixExpressionFormatter.cs @ 17622

Last change on this file since 17622 was 17622, checked in by pfleck, 4 years ago

#3040 Added vector variables to infix parser/formatter.

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