using System; using System.Collections.Generic; using System.Diagnostics; using System.Globalization; using System.Linq; using System.Diagnostics.CodeAnalysis; namespace Implab.Automaton { /// /// Алфавит символами которого являются элементы перечислений. /// /// Тип перечислений public class EnumAlphabet : IndexedAlphabetBase where T : struct, IConvertible { [SuppressMessage("Microsoft.Design", "CA1000:DoNotDeclareStaticMembersOnGenericTypes")] static readonly Lazy _symbols = new Lazy(GetSymbols); [SuppressMessage("Microsoft.Design", "CA1000:DoNotDeclareStaticMembersOnGenericTypes")] static readonly Lazy> _fullAlphabet = new Lazy>(CreateEnumAlphabet); static EnumAlphabet CreateEnumAlphabet() { var symbols = _symbols.Value; if ( symbols[symbols.Length - 1].ToInt32(CultureInfo.InvariantCulture) >= symbols.Length || symbols[0].ToInt32(CultureInfo.InvariantCulture) != 0 ) throw new InvalidOperationException("The specified enumeration must be zero-based and continuously numbered"); return new EnumAlphabet(symbols.Select(x => x.ToInt32(CultureInfo.InvariantCulture)).ToArray()); } static T[] GetSymbols() { if (!typeof(T).IsEnum) throw new InvalidOperationException("Invalid generic parameter, enumeration is required"); if (Enum.GetUnderlyingType(typeof(T)) != typeof(Int32)) throw new InvalidOperationException("Only enums based on Int32 are supported"); return ((T[])Enum.GetValues(typeof(T))) .OrderBy(x => x.ToInt32(CultureInfo.InvariantCulture)) .ToArray(); } public static EnumAlphabet FullAlphabet { get { return _fullAlphabet.Value; } } public EnumAlphabet() : base(_symbols.Value.Length) { } public EnumAlphabet(int[] map) : base(map) { Debug.Assert(map.Length == _symbols.Value.Length); } public override int GetSymbolIndex(T symbol) { return symbol.ToInt32(CultureInfo.InvariantCulture); } } }