枚举类型与位运算
目录
- 枚举
- 位运算
一、枚举
枚举类型是名称与值的组合。
枚举类型是值类型。
1、为什么枚举类型是名称与值得组合?有时我只看到键名称,没有看到相对的值。
public enum Options { Insert, Update, Save, Delete, Query }
编译器会给上面补充完整
public enum Options { Insert=0, Update=1, Save=2, Delete=3, Query=4 }
默认从0开始,依次赋值。
2、为什么枚举类型是值类型
因为System.Enum派生于System.ValueType,而System.ValueType直接派生自System.Object
编译器遇到枚举类型时,会有自己的理解:
internal struct Options:System.Enum { public const Options Insert = (Options)0; public const Options Update = (Options)1; public const Options Save = (Options)2; public const Options Delete = (Options)3; public const Options Query = (Options)4; public int value__; }
虽然Enum类型不能继承,但是可以看出编译器对待枚举类型所形成自己的理解,让我们明白其原理。
3、操作枚举类型
(1) GetUnderLyingType:返回容纳一个枚举类型的值的基础类型
每个枚举类型都有一个基础类型,默认是int.所以上面编译器默认为我们填充int类型的值0~4。
当然这个基础类型也可以自己指定,可以为byte\sbyte\short\usshot\int\uintr\long\ulong
public enum Options:byte { Insert, Update, Save, Delete, Query }
var type= Enum.GetUnderlyingType(typeof(Options)); string strs = type.ToString();
枚举类型是基元类型,我们可以对其实例进行许多平时的操作符操作,而每个枚举实例,对应着value__字段。
(2)GetValues:获取一个数组,该数组的每个元素都包含钙元素的名称和对应的值。
System.Collections.Generic.Dictionary<int,string> opsDic=new System.Collections.Generic.Dictionary<int,string>(); Options[] ops = (Options[])Enum.GetValues(typeof(Options)); foreach (var o in ops) { try { opsDic.Add((int)o, o.ToString()); } catch (Exception ex) { Console.WriteLine("键值{0}已存在", (int)o); } }
(3)GetNames:返回一组字符串名称数组。
string[] opsNames = Enum.GetNames(typeof(Options));
(4)Parse和TryParse:可以将数值和名称的字符串转换成对应的枚举类型。
Options turnOption = (Options)Enum.Parse(typeof(Options), "Insert"); Options turnOption1 = (Options)Enum.Parse(typeof(Options), "1");
Options tryTurnOption; Enum.TryParse<Options>("insert", false, out tryTurnOption);
二、位标志
我们可能需要多个枚举的组合来满足我们的需要。
位运算是针对二进制位进行的运算,常用的位运算主要有与(&), 或(|)和非(~)
e.m:1 & 0 = 0, 1 | 0 = 1, ~1 = 0
要使枚举类型具有位运算的能力,需要加上特性[FlagsAttribute]简写[Flags]。有些位处于on状态,有些处于off状态。所以通常在定义一个None=0的枚举符号。相对应的值都为2的指数倍。为的是后续的运算。
[Flags] public enum Options { None=0, Insert = 1, //二进制: 0001 Update = 2, //二进制: 0010 Save = 4, //二进制: 0100 Delete = 8, //二进制: 1000 Query = 16 //二进制:10000 }
权限列表:
Options hasOps = Options.Insert | Options.Update; string ops = hasOps.ToString();
我们看到, Options.Insert | Options.Update=0001|0010,“|”与操作:1 | 0 = 1,是二进制位的运算。我们可以得到结果为:0011。对于这个结果怎么去应用。
权限判断:
if ((hasOps & Options.Insert) == Options.Insert) { Console.WriteLine("Has {0}", Options.Insert); }
我们可以根据“&”与操作进行权限判断,可以表示为:0011&0001=0001,后面0001==0001,符合条件,有Insert权限。
实例:项目中,有时会进行正则匹配。
string[] imgsArr = Regex.Split(imgsArea, ",", RegexOptions.IgnoreCase | RegexOptions.Multiline); 这句话主要看后面:RegexOptions.IgnoreCase | RegexOptions.Multiline,可以表示为:0001|0010=0011,意思也就是需要同时符合两个条件,
:RegexOptions.IgnoreCase不区分大小写匹配 和 RegexOptions.Multiline 多行模式匹配
[Flags]
public enum RegexOptions { None = 0, IgnoreCase = 1, Multiline = 2, ExplicitCapture = 4, Compiled = 8, Singleline = 16, IgnorePatternWhitespace = 32, RightToLeft = 64, ECMAScript = 256, CultureInvariant = 512, }