首先需要从内部了解一下枚举(Enumeration),相信许多人已经知道了,当我们声明一个这样的枚举类型:
enum MyEnum
{
AAA, BBB, CCC
}
背后的IL是这样的:
.class private auto ansi sealed MyEnum
extends [mscorlib]System.Enum
{
.field public static literal valuetype Mgen.MyEnum AAA = int32(0)
.field public static literal valuetype Mgen.MyEnum BBB = int32(1)
.field public static literal valuetype Mgen.MyEnum CCC = int32(2)
.field public specialname rtspecialname int32 value__
}
那没,其实枚举中的常量都是静态的字段。而枚举对象的值会保存在非静态的特殊字段value__中。
因此,用反射来获取名称其实就是获取类型的所有静态字段就可以了,如下代码:
var fields = typeof(MyEnum).GetFields(BindingFlags.Static | BindingFlags.Public);
foreach (var fi in fields)
Console.WriteLine(fi.Name);
输出:
AAA
BBB
CCC
对于值得获取也很简单,通过反射得到的代表静态字段的FieldInfo来获取值就可以,并且获取的值仍属于枚举类型的。如果想获取枚举背后的类型,仍需要Enum.GetUnderlyingType方法,如下代码:
var fields = typeof(MyEnum).GetFields(BindingFlags.Static | BindingFlags.Public);
foreach (var fi in fields)
{
var value = fi.GetValue(null);
Console.WriteLine("值:{0} 类型:{1} 枚举背后类型:{2}",
value, value.GetType(), Enum.GetUnderlyingType(value.GetType()));
}
输出:
值:AAA 类型:Mgen.MyEnum 枚举背后类型:System.Int32
值:BBB 类型:Mgen.MyEnum 枚举背后类型:System.Int32
值:CCC 类型:Mgen.MyEnum 枚举背后类型:System.Int32
最后获取特性,通过MemberInfo类型的GetCustomAttributes方法或者Attribute类型的静态方法都可以,比如在枚举值上加一个特性:
enum MyEnum
{
AAA, BBB,
[Obsolete]
CCC
}
代码:
var fields = typeof(MyEnum).GetFields(BindingFlags.Static | BindingFlags.Public);
foreach (var fi in fields)
Console.WriteLine(Attribute.GetCustomAttribute(fi, typeof(ObsoleteAttribute), false) != null);
输出:
False
False
True