一文讀懂Python 枚舉
enum 是一組綁定到唯一常數(shù)值的符號(hào)名稱(chēng),并且具備可迭代性和可比較性的特性。我們可以使用 enum 創(chuàng)建具有良好定義的標(biāo)識(shí)符,而不是直接使用魔法字符串或整數(shù),也便于開(kāi)發(fā)工程師的代碼維護(hù)。
創(chuàng)建枚舉
我們可以使用 class 語(yǔ)法創(chuàng)建一個(gè)枚舉類(lèi)型,方便我們進(jìn)行讀寫(xiě),另外,根據(jù)函數(shù) API 的描述定義,我們可以創(chuàng)建一個(gè) enum 的子類(lèi),如下:
from enum import Enumclass HttpStatus(Enum): OK = 200 BAD_REQUEST = 400 FORBIDDEN = 403 NOT_FOUND = 404 REQUEST_TIMEOUT = 408 SERVICE_UNAVAILABLE = 500
注意: 枚舉屬性值可以是任何東西: int, str 等。如果確切的值不重要,您可以使用 auto 實(shí)例,并為您選擇適當(dāng)?shù)闹?。如果您?auto 與其他值混合,則必須小心。 枚舉類(lèi)型中,不可以設(shè)置相同名稱(chēng)的 name,可以有相同的 value。
enum 自帶屬性 name 和 value,日常工作中使用最多的也是這兩個(gè)屬性,我們打印看看結(jié)果:
print(’Member: {}’.format(HttpStatus.OK))# Member: HttpStatus.OKprint(’Member name: {}’.format(HttpStatus.OK.name)) # Member name: OKprint(’Member value: {}’.format(HttpStatus.OK.value)) # Member value: 200print(repr(HttpStatus.OK)) # <enum ’HttpStatus’>print(type(HttpStatus.OK)) # <HttpStatus.OK: 200>print(isinstance(HttpStatus.OK, HttpStatus)) # True
枚舉迭代
枚舉支持迭代和遍歷順序。舉個(gè)例子:
from enum import Enum, auto# 創(chuàng)建class HttpStatus(Enum): OK = 200 BAD_REQUEST = 400 FORBIDDEN = 403 NOT_FOUND = 404 REQUEST_TIMEOUT = 408 SERVICE_UNAVAILABLE = 500 OTHER = auto.value# 迭代for status in HttpStatus: print(’{} : {}’.format(status.name, status.value))
打印結(jié)果:
OK : 200BAD_REQUEST : 400FORBIDDEN : 403NOT_FOUND : 404REQUEST_TIMEOUT : 408SERVICE_UNAVAILABLE : 500OTHER : <object object at 0x000002863E1D7B10>
可以看出,遍歷的每一個(gè) status 是一個(gè)獨(dú)立的枚舉成員,擁有 name 和 value 屬性。
另外,我們也可以使用如下形式來(lái)進(jìn)行枚舉遍歷:
for name, member in HttpStatus.__members__.items(): print(’{} : {}’.format(name, member))
枚舉成員與屬性訪問(wèn)
通過(guò)枚舉 value 進(jìn)行訪問(wèn),訪問(wèn)需要使用元組()的形式
print(HttpStatus(200)) # HttpStatus.OK
通過(guò)枚舉 name 進(jìn)行訪問(wèn),訪問(wèn)需要使用列表[]的形式
print(HttpStatus[’OK’]) # HttpStatus.OK
將屬性賦予另一個(gè) enum 成員
number = HttpStatus.OKprint(number)# HttpStatus.OK
枚舉值唯一
上面我們創(chuàng)建的枚舉類(lèi)中,value 值是可以重復(fù)的,如果我們不想枚舉類(lèi)中的值重復(fù)可以是用裝飾器 @unique,舉例如下:
from enum import Enum, unique# 創(chuàng)建@uniqueclass HttpStatus(Enum): OK = 200 BAD_REQUEST = 400 FORBIDDEN = 403 NOT_FOUND = 404 REQUEST_TIMEOUT = 408 SERVICE_UNAVAILABLE = 500 OTHER = 200
我們運(yùn)行后,報(bào)如下異常:
ValueError: duplicate values found in <enum ’HttpStatus’>: OTHER -> OK
我們查看源代碼,發(fā)現(xiàn)加入此裝飾器的枚舉類(lèi)型,unique 方法會(huì)將其 __members__.items() 進(jìn)行遍歷,追加到 duplicates 列表中,如果發(fā)現(xiàn)列表不為空,則拋出如上異常信息。
枚舉自動(dòng)賦值
此功能用于我們?cè)谑褂妹杜e時(shí),只在意枚舉的標(biāo)識(shí)符的含義而不在意值的情況下,但是如果需要與字符串或整數(shù)混合使用就要額外注意。下面貼上官方的示例:
import unittestfrom enum import auto, Enumclass TestEnum(unittest.TestCase): def test_auto_number(self): class Color(Enum): red = auto() blue = auto() green = auto() self.assertEqual(list(Color), [Color.red, Color.blue, Color.green]) self.assertEqual(Color.red.value, 1) self.assertEqual(Color.blue.value, 2) self.assertEqual(Color.green.value, 3) def test_auto_name(self): class Color(Enum): def _generate_next_value_(self, start, count, last):return selfred = auto() blue = auto() green = auto() self.assertEqual(list(Color), [Color.red, Color.blue, Color.green]) self.assertEqual(Color.red.value, ’red’) self.assertEqual(Color.blue.value, ’blue’) self.assertEqual(Color.green.value, ’green’)
可以發(fā)現(xiàn),使用 auto() 得到的是整數(shù)自增型,如果我們需要?jiǎng)e的方式,只需要在我們的枚舉類(lèi)中,重寫(xiě) _generate_next_value_ 方法。
枚舉比較
枚舉對(duì)象可以進(jìn)行比較,但是不能進(jìn)行值比較,如果需要進(jìn)行值比較則需要枚舉對(duì)象繼承 IntEnum 對(duì)象,舉個(gè)例子:
import unittestfrom enum import Enum, IntEnumclass TestEnum(unittest.TestCase): class Season(IntEnum): SPRING = 1 SUMMER = 2 AUTUMN = 3 WINTER = 4 def test_comparisons(self): season = self.Season self.assertEqual(season.SPRING, 1) class Part(Enum): SPRING = 1 CLIP = 2 BARREL = 3 self.assertNotEqual(Part.SPRING, 1) self.assertNotEqual(Part.SPRING, season.SPRING)TestEnum().test_comparisons()
上面的測(cè)試?yán)赢?dāng)中,我們創(chuàng)建了兩個(gè)繼承類(lèi)型不一樣的枚舉類(lèi),可以看到繼承了 IntEnum 的 Season 可以進(jìn)行值的比較,而繼承了 Enum 的 Part 則不能進(jìn)行值比較,并且 IntEnum 類(lèi)型與 Enum 類(lèi)型也不能進(jìn)行比較,即使屬性和值一樣。
枚舉方法
枚舉中可以定義枚舉類(lèi)自身特有的方法,也可以復(fù)寫(xiě)一些已經(jīng)在基類(lèi)中定義好的方法,比如: __init__, __str__, __repr__ , __hash__ ,__format__ 等。舉個(gè)例子:
from enum import Enumclass Mood(Enum): FUNKY = (1, 'hello') HAPPY = (3, 'world') def describe(self): return self.name, self.value def __init__(self, num, nice): self.num = num self.nice = nice def __str__(self): return ’my custom str! {0}’.format(self.value) @classmethod def favorite_mood(cls): return cls.HAPPY @property def testValue(self): return self.nice + ’:’ + str(self.num)
上面我們定義了一個(gè)枚舉類(lèi),其中 value 是一個(gè)枚舉類(lèi)型,我們可以定義 __init__ 方法去對(duì)應(yīng)元組中的值,我們也復(fù)寫(xiě)了 __str__ 方法。
打印方法看看效果:
print(Mood.favorite_mood()) # my custom str! (3, ’world’)print(Mood.HAPPY.describe()) # (’HAPPY’, (3, ’world’))print(str(Mood.FUNKY)) # my custom str! (1, ’hello’)print(Mood.FUNKY.testValue) # hello:1
從輸出結(jié)果看,我們自定義和復(fù)寫(xiě)的方法都已經(jīng)成功的應(yīng)用到了 Mood 類(lèi)中。
枚舉繼承
不同于 java 中的枚舉類(lèi), python 中的枚舉類(lèi)是可以被繼承的,但是被繼承的枚舉類(lèi)規(guī)定其不能定義任何成員,但可以定義抽象方法。舉例如下:
class EnumExtend(unittest.TestCase): def test_extending(self): class Shade(Enum): def shade(self):print(self.name) class Color(Shade): red = 1 green = 2 blue = 3 with self.assertRaises(TypeError): class MoreColor(Color):cyan = 4magenta = 5yellow = 6 def test_extending2(self): class Shade(Enum): def shade(self):return self.name class Color(Shade): def hex(self):return ’%s nice!’ % self.value class MoreColor(Color): cyan = 4 magenta = 5 yellow = 6 self.assertEqual(MoreColor.magenta.shade(), ’magenta’) self.assertEqual(MoreColor.magenta.hex(), ’5 nice!’)
測(cè)試用例可以完美運(yùn)行,我們可以發(fā)現(xiàn):第一個(gè)方法中,拋出了 TypeError 的異常;第二個(gè)測(cè)試方法中,MoreColor 繼承了 Color, Color 繼承了 Shade, 并且我們可以通過(guò)子類(lèi)調(diào)用父類(lèi)中的方法。
總結(jié)
本節(jié)主要介紹了 enum 模塊的基礎(chǔ)知識(shí),包含枚舉的創(chuàng)建、枚舉成員和屬性的訪問(wèn)、枚舉方法的創(chuàng)建、枚舉的繼承等。其中新版中的 _ignore_、_order_、_missing_ 等可以學(xué)習(xí)官網(wǎng)的例子,另外 enum 的子類(lèi) IntEnum、IntFlag等也是我們比較常用的枚舉基類(lèi),本文中簡(jiǎn)單的介紹了 IntEnum, 而 IntFlag 相比與 IntEnum 多了 &, |, ^, ~ 的操作,其他的子類(lèi)大家感興趣也可以了解。
代碼地址
示例代碼:Python-100-days-day036
以上就是一文讀懂Python 枚舉的詳細(xì)內(nèi)容,更多關(guān)于Python 枚舉的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. ASP動(dòng)態(tài)網(wǎng)頁(yè)制作技術(shù)經(jīng)驗(yàn)分享2. jsp文件下載功能實(shí)現(xiàn)代碼3. asp.net core項(xiàng)目授權(quán)流程詳解4. 在JSP中使用formatNumber控制要顯示的小數(shù)位數(shù)方法5. CSS3實(shí)現(xiàn)動(dòng)態(tài)翻牌效果 仿百度貼吧3D翻牌一次動(dòng)畫(huà)特效6. XMLHTTP資料7. ASP常用日期格式化函數(shù) FormatDate()8. html中的form不提交(排除)某些input 原創(chuàng)9. CSS3中Transition屬性詳解以及示例分享10. ASP基礎(chǔ)入門(mén)第八篇(ASP內(nèi)建對(duì)象Application和Session)
