CLR笔记:12.枚举类型和位标志
12.1 枚举类型
强类型,枚举之间进行不能隐式转换
枚举类型直接派生于System.Enum,后者派生于System.ValueType,值类型,可以使用装箱/拆箱
不能定义方法/属性/事件。
枚举类型要与使用它的类在同一级。
默认为int,可以指定枚举成员的类型,只有int,uint,byte,sbyte,long,ulong,short,ushort这8种基元类型。
可以使用操作符作用于枚举类型,实际上是作用于相应的value实例字段,如++。
可以将枚举类型的实例显示转型为另一个不同的枚举类型,其实是先转为值,再反过去在另一个枚举中根据值找对应枚举值。
可以显示将枚举类型的实例转型为一个数值类型。
{
Red,
Black,
White
}
public enum Sentence
{
hello,
bye
}
class Program
{
static void Main(string[] args)
{
//使用运算符,++
Color a = Color.Red;
++a;
Sentence s;
//两个不同的枚举相互转换, 但是发现Color.White值为2,相应在Sentence中找不到,所以直接返回整数2
s = (Sentence)Color.White; //等价于(Sentence)1;
//两个不同的枚举相互转换,返回枚举值Black
s = (Sentence)Color.Black;
//直接输出White枚举值, 显式转为整数
Console.WriteLine(Color.White); //输出white
Console.WriteLine((Int32)Color.White); //输出2
}
}
如 enum Color{Red=1, White=2}对应IL:
可以看到,枚举只是一个在其中定义一系列常量字段和实例字段的结构。
如果只使用Color.Red这样的类型,则会作为常量处理,从而代码不再引用定义了的枚举类型,可以删除相应的dll(具体见第7章);
如果使用Color c = new Color();这样的语法,那么依赖于引用,不可以删除相应的dll
枚举方法列表:
1.GetUnderlyingType静态方法,获取枚举类型值的核心类型:
Enum.GetUnderlyingType(typeof(Color));
2.ToString()方法
Console.WriteLine(c); //输出Black
Console.WriteLine(c.ToString()); //输出Black
Console.WriteLine(c.ToString("G")); //输出Black,以上都是泛型格式
Console.WriteLine(c.ToString("D")); //输出1,十进制
Console.WriteLine(c.ToString("X")); //输出00000001,十六进制
对于c.ToString("X")十六进制格式,A-F始终是大写字母。此外,输出的数位个数取决于Color的基本类型:
byte/sbyte 2位
short/ushort 4位
int/uint 8位
long/ulong 16位
3.Format静态方法,功能基本同ToString()方法,但允许value传递一个数值,而不仅仅是一个Enum类型
4.GetValues静态方法,返回一个数组,数组中为所有的枚举类型名称:
5.static String GetName(Type enumType, Objbect value)静态方法,返回数值的字符串表示
6.static String[] GetNames(Type enumType),返回方法5的数组表示
7.Parse静态方法,将一个符号转换为枚举类型的一个实例:
Color cc2 = (Color)Enum.Parse(typeof(Color), "Black"); //可以是值,也可以是枚举类型名称
Color cc3 = (Color)Enum.Parse(typeof(Color), "Black", true); //可选参数true表示忽略大小写
8.IsDefined静态方法,判断一个数值对于一个枚举类型是否合法:
Enum.IsDefined(typeof(Color), "Black"); //返回true
常用于参数验证:
{
if (!Enum.IsDefined(typeof(Color), c))
{
//抛出异常,c不适合Color
}
}
注:Enum.IsDefined方法慎用,因为要区分大小写查找;使用反射机制,速度慢。
9.ToObject静态方法,一系列方法,如Enum.ToObject (Type, Byte),将Int32,Byte等类型实例转为枚举类型的实例(由于返回都是Object,所以要显示转型)
12.2 位标志 bit flag
位标志是一个位集合,其中有些位on,有些位off;而枚举是一个数值,是位集合中若干值的累加。
在枚举上应用[Flags]属性,就可以用枚举来表示一组可以组合的位标志,一般要定义一个值为0的None符号。
public enum FileAttributes
{
ReadOnly = 0x0001,
Hidden = 0x0002,
System = 0x0004
}
这时,就可以使用位运算来操作:
//与FileAttributes.Hidden做逻辑:如果位集合中有FileAttributes.Hidden,则返回FileAttributes.Hidden
String file = @"C:\Boot.ini";
FileAttributes attributes = File.GetAttributes(file);
if ((attributes & FileAttributes.Hidden) == FileAttributes.Hidden) { }
逻辑或,组合出所需的枚举列表,一般用来设置:
对位标记Enum调用ToString()方法,数值转字符串:
先检查是否应用了[Flags]属性
如果没有,就查找与该数值匹配的Enum符号并返回,没有匹配就返回数值。
如果有,就获取Enum降序排列的数值集合A,将A中的数值“按位与”Enum实例中的数值,将结果等于数值的Enum项附加到输出字符串,以逗号分割,如"Read,Write"
有两种特殊情况:
1.一个数值不能由Enum的若干项组合,则返回该数值
2.应该定义0。
对位标记Enum调用Parse()方法,字符串转数值:
Enum.Parse(typeof(Color), "Query, Read");
Enum.Parse(typeof(Color), "28");
处理过程如下:
1.删除字符串中所有空白字符
2.如果字符串以+/-/数字开始,那么字符串被认为是一个数字,直接转型为对应数值的字符串
3.以逗号分割字符串,在Enum中查找相应符号的数值,按位或取值并返回
不要对位标记Enum调用IsDefined()方法,因为,
传字符串,位标记中含有逗号,但永远找不到含有逗号的枚举符号;
传数值,永远返回false