Free cookie consent management tool by TermsFeed Policy Generator

source: branches/HeuristicLab.EvolutionTracking/HeuristicLab.Encodings.SymbolicExpressionTreeEncoding/3.4/SymbolicExpressionTreeNode.cs @ 11208

Last change on this file since 11208 was 11208, checked in by bburlacu, 10 years ago

#1772: Merged trunk changes.

File size: 7.7 KB
RevLine 
[3223]1#region License Information
2/* HeuristicLab
[11208]3 * Copyright (C) 2002-2014 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
[3223]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;
[4068]23using System.Collections.Generic;
[4722]24using HeuristicLab.Common;
[3223]25using HeuristicLab.Core;
26using HeuristicLab.Persistence.Default.CompositeSerializers.Storable;
27
28namespace HeuristicLab.Encodings.SymbolicExpressionTreeEncoding {
29  [StorableClass]
[5499]30  public class SymbolicExpressionTreeNode : DeepCloneable, ISymbolicExpressionTreeNode {
[3252]31    [Storable]
[5733]32    private IList<ISymbolicExpressionTreeNode> subtrees;
[3252]33    [Storable]
[5510]34    private ISymbol symbol;
[3988]35
36    // cached values to prevent unnecessary tree iterations
[5549]37    private ushort length;
38    private ushort depth;
[3988]39
[5510]40    public ISymbol Symbol {
[3484]41      get { return symbol; }
42      protected set { symbol = value; }
43    }
[3462]44
45    // parent relation is not persisted or cloned (will be set on AddSubtree or RemoveSubtree)
[5510]46    private ISymbolicExpressionTreeNode parent;
47    public ISymbolicExpressionTreeNode Parent {
[3484]48      get { return parent; }
49      set { parent = value; }
50    }
[3223]51
[4722]52    [StorableConstructor]
53    protected SymbolicExpressionTreeNode(bool deserializing) { }
54    protected SymbolicExpressionTreeNode(SymbolicExpressionTreeNode original, Cloner cloner)
[7836]55      : base(original, cloner) {
[4722]56      symbol = original.symbol; // symbols are reused
[7654]57      length = original.length;
58      depth = original.depth;
[6684]59      if (original.subtrees != null) {
60        subtrees = new List<ISymbolicExpressionTreeNode>(original.subtrees.Count);
61        foreach (var subtree in original.subtrees) {
62          var clonedSubtree = cloner.Clone(subtree);
63          subtrees.Add(clonedSubtree);
64          clonedSubtree.Parent = this;
65        }
[4722]66      }
67    }
68    public override IDeepCloneable Clone(Cloner cloner) {
69      return new SymbolicExpressionTreeNode(this, cloner);
70    }
71
72    internal SymbolicExpressionTreeNode()
73      : base() {
[3486]74      // don't allocate subtrees list here!
75      // because we don't want to allocate it in terminal nodes
76    }
[3484]77
[5510]78    public SymbolicExpressionTreeNode(ISymbol symbol)
[4722]79      : base() {
[5733]80      subtrees = new List<ISymbolicExpressionTreeNode>(3);
[3223]81      this.symbol = symbol;
82    }
83
84
[3486]85    [StorableHook(HookType.AfterDeserialization)]
[4722]86    private void AfterDeserialization() {
[5733]87      if (subtrees != null) {
88        foreach (var subtree in subtrees)
[5728]89          subtree.Parent = this;
[3486]90      }
91    }
92
[3244]93    public virtual bool HasLocalParameters {
[3223]94      get { return false; }
95    }
96
[5733]97    public virtual IEnumerable<ISymbolicExpressionTreeNode> Subtrees {
98      get { return subtrees; }
[3223]99    }
100
[5499]101    public virtual ISymbolicExpressionTreeGrammar Grammar {
[3369]102      get { return parent.Grammar; }
103    }
104
[5549]105    public int GetLength() {
106      if (length > 0) return length;
[3988]107      else {
[5549]108        length = 1;
[5733]109        if (subtrees != null) {
110          for (int i = 0; i < subtrees.Count; i++) {
111            checked { length += (ushort)subtrees[i].GetLength(); }
[4524]112          }
[4106]113        }
[5549]114        return length;
[3988]115      }
[3223]116    }
117
[5549]118    public int GetDepth() {
119      if (depth > 0) return depth;
[3988]120      else {
[5733]121        if (subtrees != null) {
122          for (int i = 0; i < subtrees.Count; i++) depth = Math.Max(depth, (ushort)subtrees[i].GetDepth());
[4106]123        }
[5549]124        depth++;
125        return depth;
[3988]126      }
[3223]127    }
[3294]128
[7506]129    public int GetBranchLevel(ISymbolicExpressionTreeNode child) {
130      return GetBranchLevel(this, child);
131    }
132
133    private static int GetBranchLevel(ISymbolicExpressionTreeNode root, ISymbolicExpressionTreeNode point) {
[11208]134      if (point == null || point.Parent == null)
135        return int.MaxValue;
136
[7506]137      if (root == point)
138        return 0;
[11208]139
140      if (root == point.Parent)
141        return 1;
142
143      var p = point.Parent;
144      int level = 1;
145
146      while (p != root) {
147        level++;
148        p = p.Parent;
149
150        if (p == null)
151          return int.MaxValue; // root is not an ancestor of point
152      }
153
154      return level;
155    }
156
157    private static int GetBranchLevelOld(ISymbolicExpressionTreeNode root, ISymbolicExpressionTreeNode point) {
[7506]158      foreach (var subtree in root.Subtrees) {
159        int branchLevel = GetBranchLevel(subtree, point);
160        if (branchLevel < int.MaxValue)
161          return 1 + branchLevel;
162      }
163      return int.MaxValue;
164    }
165
[3338]166    public virtual void ResetLocalParameters(IRandom random) { }
167    public virtual void ShakeLocalParameters(IRandom random, double shakingFactor) { }
168
[6803]169    public int SubtreeCount {
[5686]170      get {
[5733]171        if (subtrees == null) return 0;
172        return subtrees.Count;
[5686]173      }
174    }
175
[5733]176    public virtual ISymbolicExpressionTreeNode GetSubtree(int index) {
177      return subtrees[index];
[5510]178    }
[5733]179    public virtual int IndexOfSubtree(ISymbolicExpressionTreeNode tree) {
180      return subtrees.IndexOf(tree);
[5510]181    }
[5733]182    public virtual void AddSubtree(ISymbolicExpressionTreeNode tree) {
183      subtrees.Add(tree);
[3369]184      tree.Parent = this;
[3988]185      ResetCachedValues();
[3294]186    }
[5733]187    public virtual void InsertSubtree(int index, ISymbolicExpressionTreeNode tree) {
188      subtrees.Insert(index, tree);
[3369]189      tree.Parent = this;
[3988]190      ResetCachedValues();
[3294]191    }
[5733]192    public virtual void RemoveSubtree(int index) {
193      subtrees[index].Parent = null;
194      subtrees.RemoveAt(index);
[3988]195      ResetCachedValues();
[3294]196    }
197
[7795]198    public IEnumerable<ISymbolicExpressionTreeNode> IterateNodesBreadth() {
199      var list = new List<ISymbolicExpressionTreeNode>() { this };
200      int i = 0;
201      while (i != list.Count) {
202        for (int j = 0; j != list[i].SubtreeCount; ++j)
203          list.Add(list[i].GetSubtree(j));
204        ++i;
205      }
206      return list;
207    }
208
[5510]209    public IEnumerable<ISymbolicExpressionTreeNode> IterateNodesPrefix() {
210      List<ISymbolicExpressionTreeNode> list = new List<ISymbolicExpressionTreeNode>();
[3997]211      ForEachNodePrefix((n) => list.Add(n));
212      return list;
[3294]213    }
214
[5510]215    public void ForEachNodePrefix(Action<ISymbolicExpressionTreeNode> a) {
[3988]216      a(this);
[5733]217      if (Subtrees != null) {
218        foreach (var subtree in Subtrees) {
[5510]219          subtree.ForEachNodePrefix(a);
[4106]220        }
[3988]221      }
222    }
223
[5510]224    public IEnumerable<ISymbolicExpressionTreeNode> IterateNodesPostfix() {
225      List<ISymbolicExpressionTreeNode> list = new List<ISymbolicExpressionTreeNode>();
[3997]226      ForEachNodePostfix((n) => list.Add(n));
227      return list;
[3294]228    }
[3988]229
[5510]230    public void ForEachNodePostfix(Action<ISymbolicExpressionTreeNode> a) {
[5733]231      if (Subtrees != null) {
232        foreach (var subtree in Subtrees) {
[5510]233          subtree.ForEachNodePostfix(a);
[4106]234        }
[3988]235      }
236      a(this);
237    }
238
[3442]239    public override string ToString() {
[6375]240      if (Symbol != null) return Symbol.Name;
241      return "SymbolicExpressionTreeNode";
[3442]242    }
[3988]243
244    private void ResetCachedValues() {
[5549]245      length = 0; depth = 0;
[5510]246      SymbolicExpressionTreeNode parentNode = parent as SymbolicExpressionTreeNode;
247      if (parentNode != null) parentNode.ResetCachedValues();
[3988]248    }
[3223]249  }
250}
Note: See TracBrowser for help on using the repository browser.