#region License Information /* HeuristicLab * Copyright (C) 2002-2010 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * HeuristicLab is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Linq; using System.Text; using HeuristicLab.Common; using HeuristicLab.Core; using System.Xml; using HeuristicLab.Persistence.Default.CompositeSerializers.Storable; using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding.Symbols; namespace HeuristicLab.Encodings.SymbolicExpressionTreeEncoding { /// /// The default symbolic expression grammar stores symbols and syntactic constraints for symbols. /// Symbols are treated as equvivalent if they have the same name. /// Syntactic constraints limit the number of allowed sub trees for a node with a symbol and which symbols are allowed /// in the sub-trees of a symbol (can be specified for each sub-tree index separately). /// [StorableClass] [Item("DefaultSymbolicExpressionGrammar", "Represents a grammar that defines the syntax of symbolic expression trees.")] public class DefaultSymbolicExpressionGrammar : Item, ISymbolicExpressionGrammar { [Storable] private Dictionary minSubTreeCount; [Storable] private Dictionary maxSubTreeCount; [Storable] private Dictionary>> allowedChildSymbols; [Storable] private Dictionary allSymbols; public DefaultSymbolicExpressionGrammar() : base() { Reset(); } private void Initialize() { startSymbol = new StartSymbol(); AddSymbol(startSymbol); SetMinSubtreeCount(startSymbol, 1); SetMaxSubtreeCount(startSymbol, 1); } #region ISymbolicExpressionGrammar Members private Symbol startSymbol; public Symbol StartSymbol { get { return startSymbol; } set { startSymbol = value; } } protected void Reset() { minSubTreeCount = new Dictionary(); maxSubTreeCount = new Dictionary(); allowedChildSymbols = new Dictionary>>(); allSymbols = new Dictionary(); cachedMinExpressionLength = new Dictionary(); cachedMaxExpressionLength = new Dictionary(); cachedMinExpressionDepth = new Dictionary(); Initialize(); } public void AddSymbol(Symbol symbol) { if (ContainsSymbol(symbol)) throw new ArgumentException("Symbol " + symbol + " is already defined."); allSymbols.Add(symbol.Name, symbol); allowedChildSymbols[symbol.Name] = new List>(); ClearCaches(); } public void RemoveSymbol(Symbol symbol) { foreach (var parent in Symbols) { for (int i = 0; i < GetMaxSubtreeCount(parent); i++) if (IsAllowedChild(parent, symbol, i)) allowedChildSymbols[parent.Name][i].Remove(symbol.Name); } allSymbols.Remove(symbol.Name); minSubTreeCount.Remove(symbol.Name); maxSubTreeCount.Remove(symbol.Name); allowedChildSymbols.Remove(symbol.Name); ClearCaches(); } public IEnumerable Symbols { get { return allSymbols.Values.AsEnumerable(); } } public bool ContainsSymbol(Symbol symbol) { return allSymbols.ContainsKey(symbol.Name); } public void SetAllowedChild(Symbol parent, Symbol child, int argumentIndex) { if (!ContainsSymbol(parent)) throw new ArgumentException("Unknown symbol: " + parent, "parent"); if (!ContainsSymbol(child)) throw new ArgumentException("Unknown symbol: " + child, "child"); if (argumentIndex >= GetMaxSubtreeCount(parent)) throw new ArgumentException("Symbol " + parent + " can have only " + GetMaxSubtreeCount(parent) + " subtrees."); allowedChildSymbols[parent.Name][argumentIndex].Add(child.Name); ClearCaches(); } public bool IsAllowedChild(Symbol parent, Symbol child, int argumentIndex) { if (!ContainsSymbol(parent)) throw new ArgumentException("Unknown symbol: " + parent, "parent"); if (!ContainsSymbol(child)) throw new ArgumentException("Unknown symbol: " + child, "child"); if (argumentIndex >= GetMaxSubtreeCount(parent)) throw new ArgumentException("Symbol " + parent + " can have only " + GetMaxSubtreeCount(parent) + " subtrees."); return allowedChildSymbols[parent.Name][argumentIndex].Contains(child.Name); } private Dictionary cachedMinExpressionLength; public int GetMinExpressionLength(Symbol symbol) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); if (!cachedMinExpressionLength.ContainsKey(symbol.Name)) { cachedMinExpressionLength[symbol.Name] = int.MaxValue; // prevent infinite recursion long sumOfMinExpressionLengths = 1 + (from argIndex in Enumerable.Range(0, GetMinSubtreeCount(symbol)) let minForSlot = (long)(from s in Symbols where IsAllowedChild(symbol, s, argIndex) select GetMinExpressionLength(s)).DefaultIfEmpty(0).Min() select minForSlot).DefaultIfEmpty(0).Sum(); cachedMinExpressionLength[symbol.Name] = (int)Math.Min(sumOfMinExpressionLengths, int.MaxValue); } return cachedMinExpressionLength[symbol.Name]; } private Dictionary cachedMaxExpressionLength; public int GetMaxExpressionLength(Symbol symbol) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); if (!cachedMaxExpressionLength.ContainsKey(symbol.Name)) { cachedMaxExpressionLength[symbol.Name] = int.MaxValue; // prevent infinite recursion long sumOfMaxTrees = 1 + (from argIndex in Enumerable.Range(0, GetMaxSubtreeCount(symbol)) let maxForSlot = (long)(from s in Symbols where IsAllowedChild(symbol, s, argIndex) select GetMaxExpressionLength(s)).DefaultIfEmpty(0).Max() select maxForSlot).DefaultIfEmpty(0).Sum(); long limit = int.MaxValue; cachedMaxExpressionLength[symbol.Name] = (int)Math.Min(sumOfMaxTrees, limit); } return cachedMaxExpressionLength[symbol.Name]; } private Dictionary cachedMinExpressionDepth; public int GetMinExpressionDepth(Symbol symbol) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); if (!cachedMinExpressionDepth.ContainsKey(symbol.Name)) { cachedMinExpressionDepth[symbol.Name] = int.MaxValue; // prevent infinite recursion cachedMinExpressionDepth[symbol.Name] = 1 + (from argIndex in Enumerable.Range(0, GetMinSubtreeCount(symbol)) let minForSlot = (from s in Symbols where IsAllowedChild(symbol, s, argIndex) select GetMinExpressionDepth(s)).DefaultIfEmpty(0).Min() select minForSlot).DefaultIfEmpty(0).Max(); } return cachedMinExpressionDepth[symbol.Name]; } public void SetMaxSubtreeCount(Symbol symbol, int nSubTrees) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); maxSubTreeCount[symbol.Name] = nSubTrees; while (allowedChildSymbols[symbol.Name].Count <= nSubTrees) allowedChildSymbols[symbol.Name].Add(new HashSet()); while (allowedChildSymbols[symbol.Name].Count > nSubTrees) { allowedChildSymbols[symbol.Name].RemoveAt(allowedChildSymbols[symbol.Name].Count - 1); } ClearCaches(); } public void SetMinSubtreeCount(Symbol symbol, int nSubTrees) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); minSubTreeCount[symbol.Name] = nSubTrees; ClearCaches(); } public int GetMinSubtreeCount(Symbol symbol) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); return minSubTreeCount[symbol.Name]; } public int GetMaxSubtreeCount(Symbol symbol) { if (!ContainsSymbol(symbol)) throw new ArgumentException("Unknown symbol: " + symbol); return maxSubTreeCount[symbol.Name]; } #endregion private void ClearCaches() { cachedMinExpressionLength.Clear(); cachedMaxExpressionLength.Clear(); cachedMinExpressionDepth.Clear(); } public override IDeepCloneable Clone(Cloner cloner) { DefaultSymbolicExpressionGrammar clone = (DefaultSymbolicExpressionGrammar)base.Clone(cloner); clone.maxSubTreeCount = new Dictionary(maxSubTreeCount); clone.minSubTreeCount = new Dictionary(minSubTreeCount); clone.startSymbol = startSymbol; clone.allowedChildSymbols = new Dictionary>>(); foreach (var entry in allowedChildSymbols) { clone.allowedChildSymbols[entry.Key] = new List>(); foreach (var set in entry.Value) { clone.allowedChildSymbols[entry.Key].Add(new HashSet(set)); } } clone.allSymbols = new Dictionary(allSymbols); return clone; } } }