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);
}
public override IEnumerable InputSymbols {
get { return _symbols.Value; }
}
}
}