利用C#实现可以继承的”枚举”

工作中许多代码中用到枚举(enum),更用到了需要继承的枚举,由于C#的枚举不允许被继承(但允许继承自int/float等类型,这个不是我要的,在此不讨论)。

我实现了一个可以继承的模拟枚举,在此与各位分享。
于是我努力制造出可以继承的枚举,确切地说是可以继承的“仿枚举”。

首先要仿System.Enum造一个仿它的地位的类,以“操控一切”。它也是一切可继承枚举的鼻祖。

此类要承担诸多功能:

1.与int/string之间的相互转换

2.支持实例(静态属性)指定或不指定数值

3.一些辅助的功能,如比较大小等

4.一些方便使用的功能,如ForEach方法

5.像string类(class)一样,表现出值传递的效果

  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4.      namespace HeritableEnum
  5.      {
  6.          public class HEnum : IComparable<HEnum>, IEquatable<HEnum>
  7.          {
  8.              static int counter = 1; //默认数值计数器
  9.              private static Hashtable hashTable = new Hashtable(); //不重复数值集合
  10.              protected static List<HEnum> members = new List<HEnum&gt;(); //所有实例集合
  11.              private string Name { get; set; }
  12.              private int Value { get; set; }
  13.          /// <summary>
  14.              /// 不指定数值构造实例
  15.              /// </summary>
  16.              protected HEnum(string name)
  17.              {
  18.                  this.Name = name;
  19.                  this.Value = ++counter;
  20.                  members.Add(this);
  21.                  if (!hashTable.ContainsKey(this.Value))
  22.                  {
  23.                      hashTable.Add(this.Value, this);
  24.                  }
  25.              }
  26.              /// <summary>
  27.              /// 指定数值构造实例
  28.              /// </summary>
  29.              protected HEnum(string name, int value)
  30.                  : this(name)
  31.              {
  32.                  this.Value = value;
  33.                  counter = value;
  34.              }
  35.              /// <summary>
  36.              /// 向string转换
  37.              /// </summary>
  38.              /// <returns></returns>
  39.              public override string ToString()
  40.              {
  41.                  return this.Name.ToString();
  42.              }
  43.              /// <summary>
  44.              /// 显式强制从int转换
  45.              /// </summary>
  46.              /// <param name=”i”></param>
  47.              /// <returns></returns>
  48.              public static explicit operator HEnum(int i)
  49.              {
  50.                  if (hashTable.ContainsKey(i))
  51.                  {
  52.                      return (HEnum)members[i];
  53.                  }
  54.                  return new HEnum(i.ToString(), i);
  55.              }
  56.              /// <summary>
  57.              /// 显式强制向int转换
  58.              /// </summary>
  59.              /// <param name=”e”></param>
  60.              /// <returns></returns>
  61.              public static explicit operator int(HEnum e)
  62.              {
  63.                  return e.Value;
  64.              }
  65.              public static void ForEach(Action<HEnum> action)
  66.              {
  67.                  foreach (HEnum item in members)
  68.                  {
  69.                      action(item);
  70.                  }
  71.              }
  72.              public int CompareTo(HEnum other)
  73.              {
  74.                  return this.Value.CompareTo(other.Value);
  75.              }
  76.              public bool Equals(HEnum other)
  77.              {
  78.                  return this.Value.Equals(other.Value);
  79.              }
  80.              public override bool Equals(object obj)
  81.              {
  82.              if (!(obj is HEnum))
  83.                      return false;
  84.                  return this.Value == ((HEnum)obj).Value;
  85.              }
  86.          public override int GetHashCode()
  87.              {
  88.                  HEnum std = (HEnum)hashTable[this.Value];
  89.                  if (std.Name == this.Name)
  90.                      return base.GetHashCode();
  91.                  return std.GetHashCode();
  92.              }
  93.              public static bool operator !=(HEnum e1, HEnum e2)
  94.              {
  95.                  return e1.Value != e2.Value;
  96.              }
  97.              public static bool operator <(HEnum e1, HEnum e2)
  98.              {
  99.                  return e1.Value < e2.Value;
  100.              }
  101.          public static bool operator <=(HEnum e1, HEnum e2)
  102.              {
  103.                  return e1.Value <= e2.Value;
  104.          }
  105.              public static bool operator ==(HEnum e1, HEnum e2)
  106.              {
  107.                  return e1.Value == e2.Value;
  108.              }
  109.              public static bool operator >(HEnum e1, HEnum e2)
  110.              {
  111.              return e1.Value > e2.Value;
  112.              }
  113.              public static bool operator >=(HEnum e1, HEnum e2)
  114.              {
  115.                  return e1.Value >= e2.Value;
  116.              }
  117.          }
  118.      }

经过时间跨度很长中的N次尝试后,写成了上面这个样子,实现了最基本的功能。ForEach后面都是直接或间接为了“比较大小”要写的方法。

值得强调的是此类的所有构造方法必须是protected,以防止在类之外构造实例。它的子类也必须这样,以下是用于演示的子类:

  1. class EnumUse1 : HEnum
  2. {
  3.      protected EnumUse1(string name) : base(name) { }
  4.      protected EnumUse1(string name, int value) : base(name, value) { }
  5.      public static EnumUse1 A = new EnumUse1(“A”);
  6.      public static EnumUse1 B = new EnumUse1(“B”, 2);
  7.      public static EnumUse1 C = new EnumUse1(“C”, 2);
  8.      public static EnumUse1 D = new EnumUse1(“D”);
  9. }

EnumUse1从HEnum继承,模拟以下的代码

  1. enum EnumUse1
  2. {
  3.      A,
  4.      B = 2,
  5.      C = 2,
  6.      D
  7. }

再有一个子类从EnumUse1继承:

  1. class EnumUse2 : EnumUse1
  2. {
  3.      protected EnumUse2(string name) : base(name) { }
  4.      protected EnumUse2(string name, int value) : base(name, value) { }
  5.      public static EnumUse2 E = new EnumUse2(“E”);
  6. }

用起来跟系统原生的enum很像

  1. class Program
  2. {
  3.      static void Main(string[] args)
  4.      {
  5.          bool b = EnumUse1.>= EnumUse1.A;
  6.          Console.WriteLine(b.ToString());
  7.          Show(EnumUse2.E);
  8.          HEnum.ForEach((x) => Console.WriteLine(“{0} = {1},”, x, (int)x));
  9.      }
  10.      static void Show(HEnum e)
  11.      {
  12.          Console.WriteLine(@“{0} = {1},””{2}”””, e, (int)e, e.ToString());
  13.      }
  14. }

看,现在做到了可以比较大小,可以转化成string,(从string转回暂未做,但也不难),可以与int互转,值传递的效果(演示中无体现)。还比原生的enum多了ForEach功能,这点很方便。运行结果:

True
E = 4,”E”
A = 0,
B = 2,
C = 2,
D = 3,
E = 4,

话说回来,此类还有诸多不足,充其量只能算是一个实验品,想要真正走向实用,还有些工作要做。在此发布,纪念此次实验及成果。

到此这篇关于利用C#实现可以继承的”枚举”的文章就介绍到这了,更多相关C#枚举内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

标签

发表评论