Enum 指南

Enum 是一组与互不相同的值分别绑定的符号名。类似于全局变量,但提供了更好用的 repr()、分组、类型安全和一些其它特性。

它们最适用于当某个变量可选的值有限时。例如,从一周中选取一天:

  1. >>> from enum import Enum
  2. >>> class Weekday(Enum):
  3. ... MONDAY = 1
  4. ... TUESDAY = 2
  5. ... WEDNESDAY = 3
  6. ... THURSDAY = 4
  7. ... FRIDAY = 5
  8. ... SATURDAY = 6
  9. ... SUNDAY = 7

或是 RGB 三原色:

  1. >>> from enum import Enum
  2. >>> class Color(Enum):
  3. ... RED = 1
  4. ... GREEN = 2
  5. ... BLUE = 3

正如你所见,创建一个 Enum 就是简单地写一个继承 Enum 的类。

备注

枚举成员名的大小写

由于枚举被用来代表常量,并有助于避免混入类方法/属性和枚举名之间发生名称冲突,我们强烈建议用大写形式的名称表示成员,我们将在我们的示例中使用此风格。

根据枚举的性质,某个成员的值可能不一定用得上,但无论如何都能用那个值构造对应的成员:

  1. >>> Weekday(3)
  2. <Weekday.WEDNESDAY: 3>

如你所见,成员的 repr() 会显示枚举名称、成员名称和值。 成员的 str() 只会显示枚举名称和成员名称:

  1. >>> print(Weekday.THURSDAY)
  2. Weekday.THURSDAY

枚举成员的 类型 就是其所属的枚举:

  1. >>> type(Weekday.MONDAY)
  2. <enum 'Weekday'>
  3. >>> isinstance(Weekday.FRIDAY, Weekday)
  4. True

枚举成员带有一个属性,里面只是他们的 name:

  1. >>> print(Weekday.TUESDAY.name)
  2. TUESDAY

类似的,枚举成员还有一个属性 value:

  1. >>> Weekday.WEDNESDAY.value
  2. 3

很多语言只把枚举当作名称/值对,而 Python 则可以添加行为。比如 datetime.date 有两个方法用于返回工作日:weekday()isoweekday()。两者的区别是,一个是以 0-6 计数,另一个则是 1-7 计数。不用自行去作记录,只要在 Weekday 枚举中可添加一个方法,即可由 date 实例提取日期,并返回匹配的枚举成员:

  1. @classmethod
  2. def from_date(cls, date):
  3. return cls(date.isoweekday())

目前,完整的 Weekday 枚举应如下所示:

  1. >>> class Weekday(Enum):
  2. ... MONDAY = 1
  3. ... TUESDAY = 2
  4. ... WEDNESDAY = 3
  5. ... THURSDAY = 4
  6. ... FRIDAY = 5
  7. ... SATURDAY = 6
  8. ... SUNDAY = 7
  9. ... #
  10. ... @classmethod
  11. ... def from_date(cls, date):
  12. ... return cls(date.isoweekday())

现在可以知道今天是星期几了:

  1. >>> from datetime import date
  2. >>> Weekday.from_date(date.today())
  3. <Weekday.TUESDAY: 2>

当然,如果换个日子读到这篇文章,应该看到当天是周几。

如果变量只需要存一天,这个 Weekday 枚举是不错,但如果需要好几天呢?比如要写个函数来描绘一周内的家务,并且不想用 list,则可以使用不同类型的 Enum:

  1. >>> from enum import Flag
  2. >>> class Weekday(Flag):
  3. ... MONDAY = 1
  4. ... TUESDAY = 2
  5. ... WEDNESDAY = 4
  6. ... THURSDAY = 8
  7. ... FRIDAY = 16
  8. ... SATURDAY = 32
  9. ... SUNDAY = 64

这里做了两处改动:继承了 Flag,而且值都是2的幂。

就像最开始的 Weekday 枚举一样,可以只用一种类型:

  1. >>> first_week_day = Weekday.MONDAY
  2. >>> first_week_day
  3. <Weekday.MONDAY: 1>

Flag 也允许将几个成员并入一个变量:

  1. >>> weekend = Weekday.SATURDAY | Weekday.SUNDAY
  2. >>> weekend
  3. <Weekday.SATURDAY|SUNDAY: 96>

甚至可以在一个 Flag 变量上进行迭代:

  1. >>> for day in weekend:
  2. ... print(day)
  3. Weekday.SATURDAY
  4. Weekday.SUNDAY

好吧,让我们来安排家务吧:

  1. >>> chores_for_ethan = {
  2. ... 'feed the cat': Weekday.MONDAY | Weekday.WEDNESDAY | Weekday.FRIDAY,
  3. ... 'do the dishes': Weekday.TUESDAY | Weekday.THURSDAY,
  4. ... 'answer SO questions': Weekday.SATURDAY,
  5. ... }

一个显示某天家务的函数:

  1. >>> def show_chores(chores, day):
  2. ... for chore, days in chores.items():
  3. ... if day in days:
  4. ... print(chore)
  5. ...
  6. >>> show_chores(chores_for_ethan, Weekday.SATURDAY)
  7. answer SO questions

如果成员值是什么无所谓,可以少些工作,用 auto() 来取值:

  1. >>> from enum import auto
  2. >>> class Weekday(Flag):
  3. ... MONDAY = auto()
  4. ... TUESDAY = auto()
  5. ... WEDNESDAY = auto()
  6. ... THURSDAY = auto()
  7. ... FRIDAY = auto()
  8. ... SATURDAY = auto()
  9. ... SUNDAY = auto()
  10. ... WEEKEND = SATURDAY | SUNDAY

枚举成员及其属性的编程访问

有时,要在程序中访问枚举成员(如,开发时不知道颜色的确切值,Color.RED 不适用的情况)。Enum 支持如下访问方式:

  1. >>> Color(1)
  2. <Color.RED: 1>
  3. >>> Color(3)
  4. <Color.BLUE: 3>

若要用 名称 访问枚举成员时,可使用枚举项:

  1. >>> Color['RED']
  2. <Color.RED: 1>
  3. >>> Color['GREEN']
  4. <Color.GREEN: 2>

若有了枚举成员,需要获取 namevalue:

  1. >>> member = Color.RED
  2. >>> member.name
  3. 'RED'
  4. >>> member.value
  5. 1

重复的枚举成员和值

两个枚举成员的名称不能相同:

  1. >>> class Shape(Enum):
  2. ... SQUARE = 2
  3. ... SQUARE = 3
  4. ...
  5. Traceback (most recent call last):
  6. ...
  7. TypeError: 'SQUARE' already defined as 2

然而,一个枚举成员可以关联多个其他名称。如果两个枚举项 AB 具有相同值(并且首先定义的是 A ),则 B 是成员 A 的别名。对 A 按值检索将会返回成员 A。按名称检索 B 也会返回成员 A:

  1. >>> class Shape(Enum):
  2. ... SQUARE = 2
  3. ... DIAMOND = 1
  4. ... CIRCLE = 3
  5. ... ALIAS_FOR_SQUARE = 2
  6. ...
  7. >>> Shape.SQUARE
  8. <Shape.SQUARE: 2>
  9. >>> Shape.ALIAS_FOR_SQUARE
  10. <Shape.SQUARE: 2>
  11. >>> Shape(2)
  12. <Shape.SQUARE: 2>

备注

不允许创建与已定义属性(其他成员、方法等)同名的成员,也不支持创建与现有成员同名的属性。

确保枚举值唯一

默认情况下,枚举允许多个名称作为同一个值的别名。若不想如此,可以使用 unique() 装饰器:

  1. >>> from enum import Enum, unique
  2. >>> @unique
  3. ... class Mistake(Enum):
  4. ... ONE = 1
  5. ... TWO = 2
  6. ... THREE = 3
  7. ... FOUR = 3
  8. ...
  9. Traceback (most recent call last):
  10. ...
  11. ValueError: duplicate values found in <enum 'Mistake'>: FOUR -> THREE

使用自动设定的值

如果具体的枚举值无所谓是什么,可以使用 auto:

  1. >>> from enum import Enum, auto
  2. >>> class Color(Enum):
  3. ... RED = auto()
  4. ... BLUE = auto()
  5. ... GREEN = auto()
  6. ...
  7. >>> [member.value for member in Color]
  8. [1, 2, 3]

枚举值将交由 _generate_next_value_() 选取,该函数可以被重载:

  1. >>> class AutoName(Enum):
  2. ... @staticmethod
  3. ... def _generate_next_value_(name, start, count, last_values):
  4. ... return name
  5. ...
  6. >>> class Ordinal(AutoName):
  7. ... NORTH = auto()
  8. ... SOUTH = auto()
  9. ... EAST = auto()
  10. ... WEST = auto()
  11. ...
  12. >>> [member.value for member in Ordinal]
  13. ['NORTH', 'SOUTH', 'EAST', 'WEST']

备注

_generate_next_value_() 方法的定义必须在任何其他成员之前。

迭代遍历

对枚举成员的迭代遍历不会列出别名:

  1. >>> list(Shape)
  2. [<Shape.SQUARE: 2>, <Shape.DIAMOND: 1>, <Shape.CIRCLE: 3>]
  3. >>> list(Weekday)
  4. [<Weekday.MONDAY: 1>, <Weekday.TUESDAY: 2>, <Weekday.WEDNESDAY: 4>, <Weekday.THURSDAY: 8>, <Weekday.FRIDAY: 16>, <Weekday.SATURDAY: 32>, <Weekday.SUNDAY: 64>]

请注意 Shape.ALIAS_FOR_SQUAREWeekday.WEEKEND 等别名不会被显示。

特殊属性 __members__ 是一个名称与成员间的只读有序映射。包含了枚举中定义的所有名称,包括别名:

  1. >>> for name, member in Shape.__members__.items():
  2. ... name, member
  3. ...
  4. ('SQUARE', <Shape.SQUARE: 2>)
  5. ('DIAMOND', <Shape.DIAMOND: 1>)
  6. ('CIRCLE', <Shape.CIRCLE: 3>)
  7. ('ALIAS_FOR_SQUARE', <Shape.SQUARE: 2>)

__members__ 属性可用于获取枚举成员的详细信息。比如查找所有别名:

  1. >>> [name for name, member in Shape.__members__.items() if member.name != name]
  2. ['ALIAS_FOR_SQUARE']

备注

旗标的别名包括带有多个旗标设置的值,如 3,以及不设置任何旗标,即 0

比较运算

枚举成员是按 ID 进行比较的:

  1. >>> Color.RED is Color.RED
  2. True
  3. >>> Color.RED is Color.BLUE
  4. False
  5. >>> Color.RED is not Color.BLUE
  6. True

枚举值之间无法进行有序的比较。枚举的成员不是整数(另请参阅下文 IntEnum):

  1. >>> Color.RED < Color.BLUE
  2. Traceback (most recent call last):
  3. File "<stdin>", line 1, in <module>
  4. TypeError: '<' not supported between instances of 'Color' and 'Color'

相等性比较的定义如下:

  1. >>> Color.BLUE == Color.RED
  2. False
  3. >>> Color.BLUE != Color.RED
  4. True
  5. >>> Color.BLUE == Color.BLUE
  6. True

与非枚举值的比较将总是不等的(同样 IntEnum 有意设计为其他的做法,参见下文):

  1. >>> Color.BLUE == 2
  2. False

警告

重载模块是可能的 — 如果载入的模块包含枚举,它们将被重新创建,而新成员的标识号/相等性比较不一定会通过。

合法的枚举成员和属性

以上大多数示例都用了整数作为枚举值。使用整数确实简短方便(并且是 Functional API 默认提供的值),但并非强制要求。绝大多数情况下,人们并不关心枚举的实际值是什么。但如果值确实重要,可以使用任何值。

枚举是 Python 的类,可带有普通方法和特殊方法。假设有如下枚举:

  1. >>> class Mood(Enum):
  2. ... FUNKY = 1
  3. ... HAPPY = 3
  4. ...
  5. ... def describe(self):
  6. ... # self is the member here
  7. ... return self.name, self.value
  8. ...
  9. ... def __str__(self):
  10. ... return 'my custom str! {0}'.format(self.value)
  11. ...
  12. ... @classmethod
  13. ... def favorite_mood(cls):
  14. ... # cls here is the enumeration
  15. ... return cls.HAPPY
  16. ...

那么:

  1. >>> Mood.favorite_mood()
  2. <Mood.HAPPY: 3>
  3. >>> Mood.HAPPY.describe()
  4. ('HAPPY', 3)
  5. >>> str(Mood.FUNKY)
  6. 'my custom str! 1'

合法的规则如下:以单下划线开头和结尾的名称是保留值,不能使用;在枚举中定义的其他所有属性都将成为该枚举的成员,但特殊方法(__str__()__add__() 等)、描述符(方法也是描述符)和在 _ignore_ 中列出的变量名除外。

注意:如果你的枚举定义了 __new__() 和/或 __init__(),则给予枚举成员的任何值都将被传递给这些方法。 请参阅示例 Planet

备注

如果定义了 __new__() 方法,则它会在创建 Enum 成员时被使用;然后它将被 Enum 的 __new__() 所替换,该方法会在类创建后被用于查找现有成员。 详情参见 何时使用 __new__() 与 __init__()

受限的 Enum 子类化

新建的 Enum 类必须包含:一个枚举基类、至多一种数据类型和按需提供的基于 object 的混合类。这些基类的顺序如下:

  1. class EnumName([mix-in, ...,] [data-type,] base-enum):
  2. pass

仅当未定义任何成员时,枚举类才允许被子类化。因此不得有以下写法:

  1. >>> class MoreColor(Color):
  2. ... PINK = 17
  3. ...
  4. Traceback (most recent call last):
  5. ...
  6. TypeError: <enum 'MoreColor'> cannot extend <enum 'Color'>

但以下代码是可以的:

  1. >>> class Foo(Enum):
  2. ... def some_behavior(self):
  3. ... pass
  4. ...
  5. >>> class Bar(Foo):
  6. ... HAPPY = 1
  7. ... SAD = 2
  8. ...

如果定义了成员的枚举也能被子类化,则类型与实例的某些重要不可变规则将会被破坏。另一方面,一组枚举类共享某些操作也是合理的。(请参阅例程 OrderedEnum

数据类支持

当从 dataclass 继承时,__repr__() 将忽略被继承类的名称。 例如:

  1. >>> @dataclass
  2. ... class CreatureDataMixin:
  3. ... size: str
  4. ... legs: int
  5. ... tail: bool = field(repr=False, default=True)
  6. ...
  7. >>> class Creature(CreatureDataMixin, Enum):
  8. ... BEETLE = 'small', 6
  9. ... DOG = 'medium', 4
  10. ...
  11. >>> Creature.DOG
  12. <Creature.DOG: size='medium', legs=4>

使用 dataclass() 参数 repr=False 来使用标准的 repr()

在 3.12 版更改: 只有数据类字段会被显示在值区域中,而不会显示数据类的名称。Only the dataclass fields are shown in the value area, not the dataclass’ name.

打包(pickle)

枚举类型可以被打包和解包:

  1. >>> from test.test_enum import Fruit
  2. >>> from pickle import dumps, loads
  3. >>> Fruit.TOMATO is loads(dumps(Fruit.TOMATO))
  4. True

打包的常规限制同样适用于枚举类型:必须在模块的最高层级定义,因为解包操作要求可从该模块导入。

备注

用 pickle 协议版本 4 可以轻松地将嵌入其他类中的枚举进行打包。

通过在枚举类中定义 __reduce_ex__() 来修改枚举成员的封存/解封方式是可能的。 默认的方法是基于值的,但具有复杂值的枚举也许会想要基于名称的:

  1. >>> class MyEnum(Enum):
  2. ... __reduce_ex__ = enum.pickle_by_enum_name

备注

不建议为旗标使用基于名称的方式,因为未命名的别名将无法解封。

函数式 API

Enum 类可调用并提供了以下函数式 API:

  1. >>> Animal = Enum('Animal', 'ANT BEE CAT DOG')
  2. >>> Animal
  3. <enum 'Animal'>
  4. >>> Animal.ANT
  5. <Animal.ANT: 1>
  6. >>> list(Animal)
  7. [<Animal.ANT: 1>, <Animal.BEE: 2>, <Animal.CAT: 3>, <Animal.DOG: 4>]

该 API 的语义类似于 namedtuple。调用 Enum 的第一个参数是枚举的名称。

第二个参数是枚举成员名称的 来源。可以是个用空格分隔的名称字符串、名称序列、表示键/值对的二元组的序列,或者名称到值的映射(如字典)。 最后两种可以为枚举赋任意值;其他类型则会自动赋成由 1 开始递增的整数值(利用 start 形参可指定为其他起始值)。返回值是一个派生自 Enum 的新类。换句话说,上述对 Animal 的赋值等价于:

  1. >>> class Animal(Enum):
  2. ... ANT = 1
  3. ... BEE = 2
  4. ... CAT = 3
  5. ... DOG = 4
  6. ...

默认从 1 开始而非 0 ,因为 0 是布尔值 False ,但默认的枚举成员都被视作 True

对使用函数式 API 创建的枚举进行封存,可能会很棘手,因为要使用栈帧的实现细节来尝试找出枚举是在哪个模块中创建的(例如当你使用了另一个模块中的实用函数时它就可能失败,在 IronPython 或 Jython 上也可能无效)。解决办法是像下面这样显式地指定模块名称:

  1. >>> Animal = Enum('Animal', 'ANT BEE CAT DOG', module=__name__)

警告

如果未提供 module,且 Enum 无法确定是哪个模块,新的 Enum 成员将不可被解封;为了让错误尽量靠近源头,封存将被禁用。

新的 pickle 协议版本 4 在某些情况下同样依赖于 __qualname__ 被设为特定位置以便 pickle 能够找到相应的类。 例如,类是否存在于全局作用域的 SomeData 类中:

  1. >>> Animal = Enum('Animal', 'ANT BEE CAT DOG', qualname='SomeData.Animal')

完整的签名为:

  1. Enum(
  2. value='NewEnumName',
  3. names=<...>,
  4. *,
  5. module='...',
  6. qualname='...',
  7. type=<mixed-in class>,
  8. start=1,
  9. )

新枚举类将会作为名称记录的东西。

names

枚举的成员。 这可以是一个用空格或逗号分隔的字符串(值将从 1 开始除非另外指定):

  1. 'RED GREEN BLUE' | 'RED,GREEN,BLUE' | 'RED, GREEN, BLUE'

或是一个名称的迭代器对象:

  1. ['RED', 'GREEN', 'BLUE']

或是一个 (名称, 值) 对的迭代器对象:

  1. [('CYAN', 4), ('MAGENTA', 5), ('YELLOW', 6)]

或是一个映射对象:

  1. {'CHARTREUSE': 7, 'SEA_GREEN': 11, 'ROSEMARY': 42}

module — 模块

新枚举类所在的模块名。

qualname

新枚举类在模块内的限定名。

type — 类型

要并入新枚举类的类型。

start

当只传入名称时要使用的起始数值。

在 3.5 版更改: 增加了 start 形参。

派生的枚举

IntEnum

所提供的第一个变种 Enum 同时也是 int 的一个子类。 IntEnum 的成员可与整数进行比较;通过扩展,不同类型的整数枚举也可以相互进行比较:

  1. >>> from enum import IntEnum
  2. >>> class Shape(IntEnum):
  3. ... CIRCLE = 1
  4. ... SQUARE = 2
  5. ...
  6. >>> class Request(IntEnum):
  7. ... POST = 1
  8. ... GET = 2
  9. ...
  10. >>> Shape == 1
  11. False
  12. >>> Shape.CIRCLE == 1
  13. True
  14. >>> Shape.CIRCLE == Request.POST
  15. True

不过,它们仍然不可与标准 Enum 枚举进行比较:

  1. >>> class Shape(IntEnum):
  2. ... CIRCLE = 1
  3. ... SQUARE = 2
  4. ...
  5. >>> class Color(Enum):
  6. ... RED = 1
  7. ... GREEN = 2
  8. ...
  9. >>> Shape.CIRCLE == Color.RED
  10. False

IntEnum 值在其他方面的行为都如你预期的一样类似于整数:

  1. >>> int(Shape.CIRCLE)
  2. 1
  3. >>> ['a', 'b', 'c'][Shape.CIRCLE]
  4. 'b'
  5. >>> [i for i in range(Shape.SQUARE)]
  6. [0, 1]

StrEnum

所提供的第二种 Enum 变体同时也是 str 的一个子类。 StrEnum 的成员可与字符串进行比较;通过扩展,不同类型的字符串枚举也可以相互进行比较。

3.11 新版功能.

IntFlag

所提供的下一种 Enum 变体 IntFlag 也是基于 int 的。 不同之处在于 IntFlag 成员可以用位运算符 (&, |, ^, ~) 进行组合并且如果可能的话其结果仍将是 IntFlag 成员。 与 IntEnum 类似,IntFlag 成员也是整数并且可以用于任何使用 int 的地方。

备注

除位操作外,其他所有对 IntFlag 成员的操作,都会失去 IntFlag 成员资格。

导致 IntFlag 值无效的位操作将失去 IntFlag 成员资格。详见 FlagBoundary

3.6 新版功能.

在 3.11 版更改.

示例 IntFlag 类:

  1. >>> from enum import IntFlag
  2. >>> class Perm(IntFlag):
  3. ... R = 4
  4. ... W = 2
  5. ... X = 1
  6. ...
  7. >>> Perm.R | Perm.W
  8. <Perm.R|W: 6>
  9. >>> Perm.R + Perm.W
  10. 6
  11. >>> RW = Perm.R | Perm.W
  12. >>> Perm.R in RW
  13. True

对于组合同样可以进行命名:

  1. >>> class Perm(IntFlag):
  2. ... R = 4
  3. ... W = 2
  4. ... X = 1
  5. ... RWX = 7
  6. ...
  7. >>> Perm.RWX
  8. <Perm.RWX: 7>
  9. >>> ~Perm.RWX
  10. <Perm: 0>
  11. >>> Perm(7)
  12. <Perm.RWX: 7>

备注

命名的枚举组合被视作别名。别名在迭代过程中不会显示,但可以通过值查询返回。

在 3.11 版更改.

IntFlagEnum 的另一个重要区别在于如果没有设置任何旗标(值为 0),则其布尔值为 False:

  1. >>> Perm.R & Perm.X
  2. <Perm: 0>
  3. >>> bool(Perm.R & Perm.X)
  4. False

因为 IntFlag 成员也是 int 的子类,他们可以相互组合(但可能会失去 IntFlag 成员资格:

  1. >>> Perm.X | 4
  2. <Perm.R|X: 5>
  3. >>> Perm.X | 8
  4. 9

备注

否运算符 ~,始终会返回一个 IntFlag 成员的正值:

  1. >>> (~Perm.X).value == (Perm.R|Perm.W).value == 6
  2. True

IntFlag 成员也可被迭代遍历:

  1. >>> list(RW)
  2. [<Perm.R: 4>, <Perm.W: 2>]

3.11 新版功能.

标志

最后一个变体是 Flag。与 IntFlag 类似,Flag 成员可用按位运算符 (&, |, ^, ~) 组合。与 IntFlag 不同的是,它们不可与其它 Flag 枚举或 int 进行组合或比较。 虽然可以直接指定值,但推荐使用 auto 作为值来让 Flag 选择适当的值。

3.6 新版功能.

IntFlag 类似,如果 Flag 成员的某种组合导致没有设置任何旗标,则其布尔值为 False:

  1. >>> from enum import Flag, auto
  2. >>> class Color(Flag):
  3. ... RED = auto()
  4. ... BLUE = auto()
  5. ... GREEN = auto()
  6. ...
  7. >>> Color.RED & Color.GREEN
  8. <Color: 0>
  9. >>> bool(Color.RED & Color.GREEN)
  10. False

单个旗标的值应当为二的乘方 (1, 2, 4, 8, …),而旗标的组合则无此限制:

  1. >>> class Color(Flag):
  2. ... RED = auto()
  3. ... BLUE = auto()
  4. ... GREEN = auto()
  5. ... WHITE = RED | BLUE | GREEN
  6. ...
  7. >>> Color.WHITE
  8. <Color.WHITE: 7>

对 “no flags set” 条件指定一个名称并不会改变其布尔值:

  1. >>> class Color(Flag):
  2. ... BLACK = 0
  3. ... RED = auto()
  4. ... BLUE = auto()
  5. ... GREEN = auto()
  6. ...
  7. >>> Color.BLACK
  8. <Color.BLACK: 0>
  9. >>> bool(Color.BLACK)
  10. False

Flag 成员也可被迭代遍历:

  1. >>> purple = Color.RED | Color.BLUE
  2. >>> list(purple)
  3. [<Color.RED: 1>, <Color.BLUE: 2>]

3.11 新版功能.

备注

对于大多数新代码,强烈推荐使用 EnumFlag,因为 IntEnumIntFlag 打破了枚举的某些语义约定(例如可以同整数进行比较,并因而导致此行为被传递给其他无关的枚举)。 IntEnumIntFlag 的使用应当仅限于 EnumFlag 无法使用的场合;例如,当使用枚举替代整数常量时,或是与其他系统进行交互操作时。

其他事项

虽然 IntEnumenum 模块的一部分,但要独立实现也应该相当容易:

  1. class IntEnum(int, Enum):
  2. pass

这里演示了如何定义类似的派生枚举;例如一个混入了 float 而不是 intFloatEnum

几条规则:

  1. 当子类化 Enum 时,在基类序列中的混合类型必须出现于 Enum 本身之前,如以上 IntEnum 的例子所示。

  2. 混入类型必须是可子类化的。 例如,boolrange 是不可子类化的因而如果被用作混入类型就将在枚举创建期间抛出错误。

  3. 虽然 Enum 可以拥有任意类型的成员,不过一旦你混合了附加类型,则所有成员必须为相应类型的值,如在上面的例子中即为 int。 此限制不适用于仅添加方法而未指定另一数据类型的混合类。

  4. 当混合了另一数据类型时,value 属性会 不同于 枚举成员自身,但它们仍保持等价且比较结果也相等。

  5. data type 是定义了 __new__() 的混入对象,或者是一个 dataclass

  6. % 形式的格式化: %s%r 会分别调用 Enum 类的 __str__()__repr__();其他代码 (如 %i%h 用于 IntEnum) 会将枚举成员视为对应的混入类型。

  7. 格式化字符串字面值, str.format()format() 将使用枚举的 __str__() 方法。

备注

因为 IntEnum, IntFlagStrEnum 被设计为现有常量的原样替代,它们的 __str__() 方法已被重围为其数据类型的 __str__() 方法。

何时使用 __new__()__init__()

当你想要定制 Enum 成员的实际值时必须使用 __new__()。 任何其他修改可以用 __new__() 也可以用 __init__(),应优先使用 __init__()

举例来说,如果你要向构造器传入多个条目,但只希望将其中一个作为值:

  1. >>> class Coordinate(bytes, Enum):
  2. ... """
  3. ... Coordinate with binary codes that can be indexed by the int code.
  4. ... """
  5. ... def __new__(cls, value, label, unit):
  6. ... obj = bytes.__new__(cls, [value])
  7. ... obj._value_ = value
  8. ... obj.label = label
  9. ... obj.unit = unit
  10. ... return obj
  11. ... PX = (0, 'P.X', 'km')
  12. ... PY = (1, 'P.Y', 'km')
  13. ... VX = (2, 'V.X', 'km/s')
  14. ... VY = (3, 'V.Y', 'km/s')
  15. ...
  16. >>> print(Coordinate['PY'])
  17. Coordinate.PY
  18. >>> print(Coordinate(3))
  19. Coordinate.VY

警告

不要 调用 super().__new__(),因为只能找到仅用于查找的 __new__;请改为直接使用该数据类型。

细节要点

支持的 __dunder__ 名称

__members__ 是一个 member_name:member 条目的只读有序映射。 它只在类上可用。

如果指定了 __new__(),它必须创建并返回枚举成员;相应地设定成员的 _value_ 也是一个很好的主意。 一旦所有成员都创建完成它就不会再被使用。

支持的 _sunder_ 名称

  • _name_ — 成员的名称

  • _value_ — 成员的值;可以在 __new__ 中设置 / 修改

  • _missing_ — 当未发现某个值时所使用的查找函数;可被重载

  • _ignore_ — 一个名称列表,可以为 liststr,它不会被转化为成员,并将从最终类中被移除

  • _order_ — 用于 Python 2/3 代码以确保成员顺序一致(类属性,在类创建期间会被移除)

  • _generate_next_value_Functional APIauto 用它为枚举成员获取适当的值;可被重写

备注

对于标准的 Enum 类,选择的下一个值是最后所见值加1。

对于 Flag 类,下一个选择的值将是下一个最高的2次幂数,与最后所见值无关。

3.6 新版功能: _missing_, _order_, _generate_next_value_

3.7 新版功能: _ignore_

用来帮助 Python 2 / Python 3 代码保持同步提供 _order_ 属性。 它将与枚举的实际顺序进行对照检查,如果两者不匹配则会引发错误:

  1. >>> class Color(Enum):
  2. ... _order_ = 'RED GREEN BLUE'
  3. ... RED = 1
  4. ... BLUE = 3
  5. ... GREEN = 2
  6. ...
  7. Traceback (most recent call last):
  8. ...
  9. TypeError: member order does not match _order_:
  10. ['RED', 'BLUE', 'GREEN']
  11. ['RED', 'GREEN', 'BLUE']

备注

在 Python 2 代码中 _order_ 属性是必须的,因为定义顺序在被记录之前就会丢失。

_Private__names

私有名称 不会被转换为枚举成员,而是保持为普通属性。

在 3.11 版更改.

Enum 成员类型

枚举成员是其枚举类的实例,并且通常以 EnumClass.member 的形式来访问。 在特定场景下,如编写自定义枚举行为,可直接从一个成员访问另一个成员的能力是很有用的,并且是受支持的;但是,为了避免成员名与混入类属性/方法之间发生名称冲突,强烈建议使用大写形式的名称。

在 3.5 版更改.

创建与其他数据类型混合的成员

当使用 Enum 来子类化其他数据类型,如 intstr 时,所有在 = 之后的值都会被传递给该数据类型的构造器。 例如:

  1. >>> class MyEnum(IntEnum): # help(int) -> int(x, base=10) -> integer
  2. ... example = '11', 16 # so x='11' and base=16
  3. ...
  4. >>> MyEnum.example.value # and hex(11) is...
  5. 17

Enum 类和成员的布尔值

与非 Enum 类型(如 intstr 等)混合的枚举类会根据混合类型的规则进行计算;否则,所有成员都计算为 True。为了使你自己的枚举的布尔值取决于成员的值,请在你的类中添加以下内容:

  1. def __bool__(self):
  2. return bool(self.value)

普通的 Enum 类总是计算为 True

带有方法的 Enum

如果你给你的枚举子类提供了额外的方法,如下面的 Planet 类那样,这些方法将显示在成员的,而不是类的 dir() 中:

  1. >>> dir(Planet)
  2. ['EARTH', 'JUPITER', 'MARS', 'MERCURY', 'NEPTUNE', 'SATURN', 'URANUS', 'VENUS', '__class__', '__doc__', '__members__', '__module__']
  3. >>> dir(Planet.EARTH)
  4. ['__class__', '__doc__', '__module__', 'mass', 'name', 'radius', 'surface_gravity', 'value']

组合 Flag 的成员

遍历 Flag 成员的组合将只返回由一个比特组成的成员:

  1. >>> class Color(Flag):
  2. ... RED = auto()
  3. ... GREEN = auto()
  4. ... BLUE = auto()
  5. ... MAGENTA = RED | BLUE
  6. ... YELLOW = RED | GREEN
  7. ... CYAN = GREEN | BLUE
  8. ...
  9. >>> Color(3) # named combination
  10. <Color.YELLOW: 3>
  11. >>> Color(7) # not named combination
  12. <Color.RED|GREEN|BLUE: 7>

FlagIntFlag 的细节

使用以下代码段作为我们的例子:

  1. >>> class Color(IntFlag):
  2. ... BLACK = 0
  3. ... RED = 1
  4. ... GREEN = 2
  5. ... BLUE = 4
  6. ... PURPLE = RED | BLUE
  7. ... WHITE = RED | GREEN | BLUE
  8. ...

下列情况为True:

  • 单比特标志是典型的

  • 多比特和零比特标志是别名

  • 迭代过程中只返回典型的标志:

    1. >>> list(Color.WHITE)
    2. [<Color.RED: 1>, <Color.GREEN: 2>, <Color.BLUE: 4>]
  • 取负一个标志或标志集会返回一个新的标志/标志集和其对应的正整数值:

    1. >>> Color.BLUE
    2. <Color.BLUE: 4>
    3. >>> ~Color.BLUE
    4. <Color.RED|GREEN: 3>
  • 伪标志的名称是由其成员的名称构建的:

    1. >>> (Color.RED | Color.GREEN).name
    2. 'RED|GREEN'
  • 多位标志,又称别名,可以从操作中返回:

    1. >>> Color.RED | Color.BLUE
    2. <Color.PURPLE: 5>
    3. >>> Color(7) # or Color(-1)
    4. <Color.WHITE: 7>
    5. >>> Color(0)
    6. <Color.BLACK: 0>
  • 成员 / 包含检测:零值旗标总是会被视为包含:

    1. >>> Color.BLACK in Color.WHITE
    2. True

    在其他情况下,仅当一个旗标的所有比特位都包含于另一个旗标中才会返回 True:

    1. >>> Color.PURPLE in Color.WHITE
    2. True
    3. >>> Color.GREEN in Color.PURPLE
    4. False

有一个新的边界机制,控制如何处理超出范围的/无效的比特:STRICTCONFORMEJECTKEEP

  • STRICT —> 当出现无效的值时,会触发一个异常。

  • CONFORM —> 丢弃任何无效的比特

  • EJECT —> 失去Flag的状态,成为一个普通的int,其值为给定值。

  • KEEP —> 保留额外的比特

    • 保留Flag状态和额外的比特

    • 额外的比特不会在迭代中显示出来

    • 在repr()和str()中确实显示了额外的比特

默认的标志为 STRICTIntFlag``默认为``EJECT_convert_``默认为``KEEP``(需要``KEEP``的例子见``ssl.Options)。

枚举和旗标有何差异?

Enum有一个自定义的元类,它影响到派生的 Enum 类和它们的实例(成员)的许多方面。

枚举类

EnumType 元类负责提供 __contains__(), __dir__(), __iter__() 和其他方法来允许人们在 Enum 类上做一些在普通类上会失败的事情,比如 list(Color)some_enum_var in Color 等。 EnumType 负责确保最终的 Enum 类上的各种其他方法是正确的(比如 __new__(), __getnewargs__(), __str__()__repr__() 等)。

旗标类

旗标具有扩展的别名视图:为了符合规范,旗标的值必须为二的乘方,且名称不可重复。 因此,除了别名的定义 Enum 之外,没有值 (即 0) 或是几个二的乘方值之和 (如 3) 的旗标也会被视为别名。

枚举成员(即实例)

EnumType 在创建枚举类本身的时候,创建了所有的枚举类,然后把一个自定义的 __new__() 放在那里,通过只返回现有的成员实例来确保没有新的成员被实例化。

旗标成员

旗标成员可以如 Flag 类一样被迭代,并且只有规范的成员会被返回。 例如:

  1. >>> list(Color)
  2. [<Color.RED: 1>, <Color.GREEN: 2>, <Color.BLUE: 4>]

(请注意 BLACK, PURPLEWHITE 将不显示。)

对一个旗标成员取反将返回对应的正值,而不是负值 —- 例如:

  1. >>> ~Color.RED
  2. <Color.GREEN|BLUE: 6>

旗标成员具有与它们所包含的二的乘方值的数量相对应的长度。 例如:

  1. >>> len(Color.PURPLE)
  2. 2

枚举指导手册

虽然 Enum, IntEnum, StrEnum, FlagIntFlag 有望能涵盖大多数的使用情况,但它们不能涵盖所有情况。 这里有一些不同类型的枚举的方法,可以直接使用,或者作为创建定制枚举的范例。

省略值

在许多应用场景中,人们并不关心枚举的实际值是什么。 有几种方式可用来定义这种类型的简单枚举:

  • 使用 auto 的实例作为值

  • 使用 object 的实例作为值

  • 使用描述性的字符串作为值

  • 使用元组作为值并用自定义的 __new__() 以一个 int 值来替代该元组

使用以上任何一种方法均可向用户指明值并不重要,并且使人能够添加、移除或重排序成员而不必改变其余成员的数值。

使用 auto

使用 auto 的形式如下:

  1. >>> class Color(Enum):
  2. ... RED = auto()
  3. ... BLUE = auto()
  4. ... GREEN = auto()
  5. ...
  6. >>> Color.GREEN
  7. <Color.GREEN: 3>

使用 object

使用 object 的形式如下:

  1. >>> class Color(Enum):
  2. ... RED = object()
  3. ... GREEN = object()
  4. ... BLUE = object()
  5. ...
  6. >>> Color.GREEN
  7. <Color.GREEN: <object object at 0x...>>

这也是一个可以说明为什么你会需要编写自己的 __repr__() 的好例子:

  1. >>> class Color(Enum):
  2. ... RED = object()
  3. ... GREEN = object()
  4. ... BLUE = object()
  5. ... def __repr__(self):
  6. ... return "<%s.%s>" % (self.__class__.__name__, self._name_)
  7. ...
  8. >>> Color.GREEN
  9. <Color.GREEN>

使用描述性字符串

使用字符串作为值的形式如下:

  1. >>> class Color(Enum):
  2. ... RED = 'stop'
  3. ... GREEN = 'go'
  4. ... BLUE = 'too fast!'
  5. ...
  6. >>> Color.GREEN
  7. <Color.GREEN: 'go'>

使用自定义的 __new__()

使用自动编号 __new__() 的形式如下:

  1. >>> class AutoNumber(Enum):
  2. ... def __new__(cls):
  3. ... value = len(cls.__members__) + 1
  4. ... obj = object.__new__(cls)
  5. ... obj._value_ = value
  6. ... return obj
  7. ...
  8. >>> class Color(AutoNumber):
  9. ... RED = ()
  10. ... GREEN = ()
  11. ... BLUE = ()
  12. ...
  13. >>> Color.GREEN
  14. <Color.GREEN: 2>

要实现更通用的 AutoNumber,请添加 *args 到签名中:

  1. >>> class AutoNumber(Enum):
  2. ... def __new__(cls, *args): # this is the only change from above
  3. ... value = len(cls.__members__) + 1
  4. ... obj = object.__new__(cls)
  5. ... obj._value_ = value
  6. ... return obj
  7. ...

这样当你从 AutoNumber 继承时你将可以编写你自己的 __init__ 来处理任何附加参数:

  1. >>> class Swatch(AutoNumber):
  2. ... def __init__(self, pantone='unknown'):
  3. ... self.pantone = pantone
  4. ... AUBURN = '3497'
  5. ... SEA_GREEN = '1246'
  6. ... BLEACHED_CORAL = () # New color, no Pantone code yet!
  7. ...
  8. >>> Swatch.SEA_GREEN
  9. <Swatch.SEA_GREEN: 2>
  10. >>> Swatch.SEA_GREEN.pantone
  11. '1246'
  12. >>> Swatch.BLEACHED_CORAL.pantone
  13. 'unknown'

备注

如果定义了 __new__() 则它会在创建 Enum 成员期间被使用;随后它将被 Enum 的 __new__() 所替换,该方法会在类创建后被用来查找现有成员。

警告

不要 调用 super().__new__(),因为只能找到仅用于查找的 __new__;请改为直接使用该数据类型 — 例如:

  1. obj = int.__new__(cls, value)

OrderedEnum

一个有序枚举,它不是基于 IntEnum,因此保持了正常的 Enum 不变特性(例如不可与其他枚举进行比较):

  1. >>> class OrderedEnum(Enum):
  2. ... def __ge__(self, other):
  3. ... if self.__class__ is other.__class__:
  4. ... return self.value >= other.value
  5. ... return NotImplemented
  6. ... def __gt__(self, other):
  7. ... if self.__class__ is other.__class__:
  8. ... return self.value > other.value
  9. ... return NotImplemented
  10. ... def __le__(self, other):
  11. ... if self.__class__ is other.__class__:
  12. ... return self.value <= other.value
  13. ... return NotImplemented
  14. ... def __lt__(self, other):
  15. ... if self.__class__ is other.__class__:
  16. ... return self.value < other.value
  17. ... return NotImplemented
  18. ...
  19. >>> class Grade(OrderedEnum):
  20. ... A = 5
  21. ... B = 4
  22. ... C = 3
  23. ... D = 2
  24. ... F = 1
  25. ...
  26. >>> Grade.C < Grade.A
  27. True

DuplicateFreeEnum

如果发现重复的成员名称则会引发一个错误而不是创建一个别名:

  1. >>> class DuplicateFreeEnum(Enum):
  2. ... def __init__(self, *args):
  3. ... cls = self.__class__
  4. ... if any(self.value == e.value for e in cls):
  5. ... a = self.name
  6. ... e = cls(self.value).name
  7. ... raise ValueError(
  8. ... "aliases not allowed in DuplicateFreeEnum: %r --> %r"
  9. ... % (a, e))
  10. ...
  11. >>> class Color(DuplicateFreeEnum):
  12. ... RED = 1
  13. ... GREEN = 2
  14. ... BLUE = 3
  15. ... GRENE = 2
  16. ...
  17. Traceback (most recent call last):
  18. ...
  19. ValueError: aliases not allowed in DuplicateFreeEnum: 'GRENE' --> 'GREEN'

备注

这个例子适用于子类化 Enum 来添加或改变禁用别名以及其他行为。 如果需要的改变只是禁用别名,也可以选择使用 unique() 装饰器。

Planet

如果定义了 __new__()__init__(),则枚举成员的值将被传给这些方法:

  1. >>> class Planet(Enum):
  2. ... MERCURY = (3.303e+23, 2.4397e6)
  3. ... VENUS = (4.869e+24, 6.0518e6)
  4. ... EARTH = (5.976e+24, 6.37814e6)
  5. ... MARS = (6.421e+23, 3.3972e6)
  6. ... JUPITER = (1.9e+27, 7.1492e7)
  7. ... SATURN = (5.688e+26, 6.0268e7)
  8. ... URANUS = (8.686e+25, 2.5559e7)
  9. ... NEPTUNE = (1.024e+26, 2.4746e7)
  10. ... def __init__(self, mass, radius):
  11. ... self.mass = mass # in kilograms
  12. ... self.radius = radius # in meters
  13. ... @property
  14. ... def surface_gravity(self):
  15. ... # universal gravitational constant (m3 kg-1 s-2)
  16. ... G = 6.67300E-11
  17. ... return G * self.mass / (self.radius * self.radius)
  18. ...
  19. >>> Planet.EARTH.value
  20. (5.976e+24, 6378140.0)
  21. >>> Planet.EARTH.surface_gravity
  22. 9.802652743337129

TimePeriod

一个演示如何使用 _ignore_ 属性的例子:

  1. >>> from datetime import timedelta
  2. >>> class Period(timedelta, Enum):
  3. ... "different lengths of time"
  4. ... _ignore_ = 'Period i'
  5. ... Period = vars()
  6. ... for i in range(367):
  7. ... Period['day_%d' % i] = i
  8. ...
  9. >>> list(Period)[:2]
  10. [<Period.day_0: datetime.timedelta(0)>, <Period.day_1: datetime.timedelta(days=1)>]
  11. >>> list(Period)[-2:]
  12. [<Period.day_365: datetime.timedelta(days=365)>, <Period.day_366: datetime.timedelta(days=366)>]

子类化 EnumType

虽然大多数枚举需求可以通过自定义 Enum 子类来满足,无论是用类装饰器还是自定义函数,EnumType 可以被子类化以提供不同的枚举体验。