C#中枚舉的特性 FlagAttribute詳解
寫在前面
枚舉Enum 全稱(Enumeration),即一種由一組稱為枚舉數(shù)列表的命名常量組成的獨(dú)特類型??梢钥闯雒杜e的出現(xiàn)時(shí)為了使我們可以在程序中方便的使用一些特定值的常量,一般的使用大家都比較熟悉,本文主要介紹枚舉的特性 FlagAttribute。
FlagAttribute是什么?
Flag 特性微軟的解釋是:指示可以將枚舉作為位域(即一組標(biāo)志)處理,F(xiàn)lagsAttribute屬性就是枚舉類型的一項(xiàng)可選屬性,它的主要作用是可以將枚舉作為位域處理(P.S. C#不支持位域)。所謂位域是單個(gè)存儲(chǔ)單元內(nèi)相鄰二進(jìn)制位的集合。通過為枚舉添加這個(gè)屬性,可以改變枚舉的一些行為來滿足我們的需要。
比如我們有如下枚舉的定義:
public enum OrderTypeEnum { Init, Complete, Waiting, Paid }
邏輯或操作我相信大家都比較熟悉了,對于整數(shù)來說,| 操作就是將其轉(zhuǎn)化為二進(jìn)制再進(jìn)行或運(yùn)算。OrderTypeEnum.Init | OrderTypeEnum.Complete做的工作實(shí)際上是 0001 | 0010 = 0011 = 3再轉(zhuǎn)換成(OrderTypeEnum)3就是OrderTypeEnum.Paid了.
如果我們對兩個(gè)枚舉值做 | 操作,那結(jié)果會(huì)是什么樣呢?
OrderTypeEnum result = OrderTypeEnum.Waiting | OrderTypeEnum.Paid;
按照或操作的原理:0010 | 0011 = 0011(3) Paid ,實(shí)質(zhì)上我們想要的結(jié)果是想講兩個(gè)枚舉值都作為或操作的結(jié)果,但是因?yàn)槊杜e值默認(rèn)是從0開始順次遞增的,那么經(jīng)過或操作之后就得不到我們想要的結(jié)果,那怎么辦呢,這時(shí)候就需要 給枚舉加上 [Flags] 的Attribute,我們先來看一下FlagsAttribute定義的準(zhǔn)則:
- 使用FlagsAttribute枚舉才是對數(shù)字值執(zhí)行按位運(yùn)算 (AND、 OR 獨(dú)占或) 的自定義屬性。
- 在 2 的冪,即 1、 2、 4、 8 等中定義枚舉常量。 這意味著不重疊中組合的枚舉常量的各個(gè)標(biāo)志。
- 請考慮創(chuàng)建針對常用的標(biāo)志組合的枚舉的常數(shù)。 例如,如果你有用于文件 I/O 操作的枚舉包含枚舉的常數(shù)Read = 1和Write = 2,請考慮創(chuàng)建枚舉的常數(shù)ReadWrite = Read OR Write,它結(jié)合Read和Write標(biāo)志。 此外,可用于組合標(biāo)志的按位 OR 操作視為在某些情況下,不應(yīng)為用于簡單任務(wù)所需的一個(gè)高級(jí)的概念。
- 如果為標(biāo)志枚舉常量中定義為負(fù)數(shù),因?yàn)楹芏鄻?biāo)志位置可能會(huì)設(shè)置為 1,這可能會(huì)使你的代碼的混亂,并鼓勵(lì)編碼錯(cuò)誤,請務(wù)必小心。
- 測試是否在數(shù)值中設(shè)置一個(gè)標(biāo)志一種簡便方式是執(zhí)行按位,操作之間的數(shù)字值和標(biāo)志枚舉的常數(shù),它將所有位都設(shè)置為不對應(yīng)于標(biāo)志的零的數(shù)字值中,然后測試該操作的結(jié)果是否等于該標(biāo)志枚舉常量。
- 使用None用作枚舉其值為零的常量的標(biāo)志名稱。 不能使用None按位運(yùn)算中,來測試一個(gè)標(biāo)志,因?yàn)榻Y(jié)果始終為零的枚舉的常數(shù)。 但是,你可以執(zhí)行的邏輯不之間的數(shù)字值的按位、 比較和None枚舉的常量,以確定是否已設(shè)置在數(shù)值中的任何位。
- 如果你創(chuàng)建而不是標(biāo)志枚舉的值枚舉,它是仍必要?jiǎng)?chuàng)建None枚舉的常數(shù)。 原因是,默認(rèn)情況下用于枚舉的內(nèi)存初始化為零的公共語言運(yùn)行時(shí)。 因此,如果未定義其值為零的常量,枚舉將包含在創(chuàng)建時(shí)非法值。
- 如果你的應(yīng)用程序需要表示明顯默認(rèn)情況下,請考慮使用其值為零表示默認(rèn)值的枚舉的常數(shù)。 如果沒有任何默認(rèn)情況下,請考慮使用其值為零的枚舉的常數(shù)意味著不由任何其他枚舉常量表示這種情況。
- 未定義一個(gè)枚舉值,只是為了鏡像與枚舉本身的狀態(tài)。 例如,不定義僅用于枚舉的結(jié)束標(biāo)記的枚舉的常數(shù)。 如果你需要確定在枚舉的最后一個(gè)值,請顯式檢查該值。 此外,你可以執(zhí)行范圍檢查第一個(gè)和最后一個(gè)枚舉常量,如果范圍內(nèi)的所有值都是有效。
- 不要指定保留供將來使用的枚舉的常數(shù)。
- 當(dāng)你定義的方法或?qū)傩?,它采用作為值的枚舉的常數(shù)時(shí),請考慮驗(yàn)證值。 原因是,即使該數(shù)值不在枚舉中定義,你可以強(qiáng)制轉(zhuǎn)換為枚舉類型的數(shù)字值。
我們看到第二句告訴我們當(dāng)加了Flags的特性之后默認(rèn)的枚舉值就會(huì)以2的冪一次遞增,比如 20,21,22,23(1,2,4,8....)
那我們重新看一下重新定義之后的或操作會(huì)是什么結(jié)果呢?
[Flags] public enum OrderTypeEnum { Init, Complete, Waiting, Paid }
此時(shí)我們再來看:OrderTypeEnum result = OrderTypeEnum.Complete | OrderTypeEnum.Waiting | OrderTypeEnum.Paid ;
0010 | 0100 | 1000 = 1110 我們可以看到實(shí)質(zhì)上就是做了二進(jìn)制的或運(yùn)算,將所有位值做了合并
當(dāng)我們可以用做位運(yùn)算的時(shí)候,就不僅僅是或,與,非,異或等操作都可以實(shí)現(xiàn)。
我們知道通過這樣可以把枚舉值合并 OrderTypeEnum result = OrderTypeEnum.Complete | OrderTypeEnum.Waiting | OrderTypeEnum.Paid ;
那么同理也可以來判斷這樣的集合中是否包含某個(gè)枚舉值:
result.HasFlag(OrderTypeEnum.Paid)
寫在最后
枚舉通過添加Flags的特性使得它能夠擁有位運(yùn)算的能力,更方便了我們再日常代碼中的使用。
參考資料:http://www.alanzucconi.com/2015/07/26/enum-flags-and-bitwise-operators/
相關(guān)文章
DevExpress設(shè)置餅狀圖的Lable位置實(shí)例
這篇文章主要介紹了DevExpress設(shè)置餅狀圖的Lable位置的方法,以實(shí)例形式詳細(xì)講述了設(shè)置餅狀圖的Lable位置具體實(shí)現(xiàn)過程,需要的朋友可以參考下2014-10-10Repeater中添加按鈕實(shí)現(xiàn)點(diǎn)擊按鈕獲取某一行數(shù)據(jù)的方法
這篇文章主要介紹了Repeater中添加按鈕實(shí)現(xiàn)點(diǎn)擊按鈕獲取某一行數(shù)據(jù)的方法,是非常實(shí)用的一個(gè)技巧,需要的朋友可以參考下2014-08-08unity 文件流讀取圖片與www讀取圖片的區(qū)別介紹
這篇文章主要介紹了unity 文件流讀取圖片與www讀取圖片的對比分析,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-04-04C#采用FileSystemWatcher實(shí)現(xiàn)監(jiān)視磁盤文件變更的方法
這篇文章主要介紹了C#采用FileSystemWatcher實(shí)現(xiàn)監(jiān)視磁盤文件變更的方法,詳細(xì)分析了FileSystemWatcher的用法,并以此為基礎(chǔ)實(shí)現(xiàn)監(jiān)視磁盤文件變更,是非常實(shí)用的技巧,具有一定的借鑒價(jià)值,需要的朋友可以參考下2014-11-11