C# 中枚举的一点研究(跳过一些net坑的研究而已)

时间:2023-03-09 06:27:00
C# 中枚举的一点研究(跳过一些net坑的研究而已)

之前一直使用Enum.Parse()将字符串转为枚举,没有深究,后面发现一个问题后对下面的Enum有了一个初步研究(.net 4.0).

枚举是一个指定的常数集,其基础类型可以是除 Char 外的任何整型。 如果没有显式声明基础类型,则使用 Int32。 Enum 是 .NET中所有枚举的基类。

永远不要创建基础类型为非整形的枚举类型。 虽然可以通过使用反射来创建这一枚举类型,但使用得出的类型的方法调用是不可靠的,也可能会引发其他异常。

看下面代码.

(留意,枚举类型是值类型,其值不能为Null,所以我们定义一个枚举类型,其默认值通常为该枚举的第一个元素,或值为0的元素)

首先定义一个test枚举

enum test { aaa,bbb,ccc,ddd}

(1)现在对比Parse方法(留意,parse方法里面value参数是区分大小写的,如果想忽略大小写,一定要使用带ignoreCase参数的覆写方法)

private void button1_Click(object sender, EventArgs e)
{
test te;
string typeStr; te = (test)Enum.Parse(typeof(test), "4"); //4
typeStr = te.GetType().Name; //test
    te = (test)Enum.Parse(typeof(test), "2"); //ccc
typeStr = te.GetType().Name; //test     te = (test)Enum.Parse(typeof(test), "ddd"); //ddd
    te = (test)Enum.Parse(typeof(test), "ddd1"); //System.ArgumentException("未找到请求的值“ddd1”。")
}
那么看看TryParse方法呢? //(留意,TryParse方法未经初始化即被赋值为默认值,如果不想TryParse方法改变Out的枚举值,先调用 IsDefined()(IsDefined对于字符串区分大小写的)方法
//以确保整数的特定字符串表示形式实际是 TEnum 的成员。)
test te = test.bbb; 
if (Enum.TryParse("aaa", out te)) //aaa 返回值为True
 MessageBox.Show(te.GetType().Name); //test
if (Enum.TryParse("111", out te)) //111 返回值为False
    MessageBox.Show(te.GetType().Name);
TryParse方法的定义:
public static bool TryParse<TEnum>(
string value,
out TEnum result
)
where TEnum : struct
TryParse方法有点奇特,只要是value能转换为数字,那么一定会成功,如果不能,则根据大小写(默认区分大小写)去转换.不存在就返回false.

结论,对于Enum.Parse() 方法,即使是想通过TryParse()方法判断返回的bool值,也建议首先使用Enum.IsDefined()判断是否存在.

(2)Enum.ToObject()方法

te = (test)Enum.ToObject(typeof(test), 4); //4
由此可见,ToObject()也同Parse方法一样,对于数值,都不会引发ArgumentException,而转换成功,所以也建议首先使用Enum.IsDefined()判断是否存在.

(3)Enum.GetValues()方法

  这个方法并不能获取结果为{0,1,2,3}的数组,而是返回和GetNames()方法一样的,只不过GetNames返回值类型为string[].
如果想获取{0,1,2,3},参考这个方法

public static Array getValuesArray(Type enumType)
{
    Array tempArry;
    tempArry = Enum.GetValues(enumType);
    //tempArry = enumType.GetEnumValues();
    Array result = Array.CreateInstance(typeof(int), tempArry.Length);
    int j = 0;
    foreach (int i in tempArry)
    {
        result.SetValue(i, j++);
    }
    return result;
 }
(4) 一直说用Enum.IsDefined()方法判断是否存在,那么还是看看这个方法吧.
bool b;
b = Enum.IsDefined(typeof(test), 3);//True
b = Enum.IsDefined(typeof(test), 4);//false
b = Enum.IsDefined(typeof(test), "aaa");//true
b = Enum.IsDefined(typeof(test), "aaA");//false
b = Enum.IsDefined(typeof(test), "aaA1");//false
(5)现在在看看Enum.GetName(Type enumType, Object value)
方法对于字符串和整形是怎么样处理的
string s;
s = Enum.GetName(typeof(test), 3);//ddd
s = Enum.GetName(typeof(test), 4);//null
//s = Enum.GetName(typeof(test), "3");//System.ArgumentException(传入的值必须是枚举的枚举基或基础类型,如 Int32)            
//s = Enum.GetName(typeof(test), "aaa");//System.ArgumentException(传入的值必须是枚举的枚举基或基础类型,如 Int32)
 
基于以上的总结,以后但凡是想将字符串或整形转为枚举,那么首先用Enum.IsDefined()方法判断,
对于字符串,用Parse或TryParse转换,由于.net方法太多常见是不区分大小写的,所以记得使用带ignoreCase参数的覆写方法.
对于整形,可以使用Parse和ToObject方法.将整数转换为枚举值时,可以分配一个实际并非枚举成员的值。 为防止该情况,可在执行转换前将该整数传递至 IsDefined() 方法。
将枚举转为字符串或整形值,可使用可通过调用静态 Format 方法以及重载实例 ToString()方法.或GetName().
Enum 类提供 IConvertible 接口的显式接口实现以从枚举值转换为整数类型,所以可以使用Convert.ToInt32.
Convert.ToInt32(test.aaa);
如果想将枚举元素或(值)转为字符串(整形),可用Enum.GetValues用(int)转换一下.