百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

python 系列(枚举类型)

bigegpt 2024-09-09 01:23 51 浏览

枚举 - 枚举类型

该enum模块定义了具有迭代和比较功能的枚举类型。它可用于为值创建定义明确的符号,而不是使用文字整数或字符串。

创建枚举

class通过子类化Enum和添加描述值的类属性,使用语法 定义新的枚举。

enum_create.py

import enum
class BugStatus(enum.Enum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
print('\nMember name: {}'.format(BugStatus.wont_fix.name))
print('Member value: {}'.format(BugStatus.wont_fix.value))

在Enum解析类时,将成员转换为实例。每个实例都具有name与成员名称value对应的属性以及与在类定义中分配给名称的值对应的属性。

$ python3 enum_create.py
Member name: wont_fix
Member value: 4

迭代

迭代枚举会产生枚举的各个成员。

enum_iterate.py

import enum
class BugStatus(enum.Enum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
for status in BugStatus:
 print('{:15} = {}'.format(status.name, status.value))

成员按照在类定义中声明的顺序生成。名称和值不用于以任何方式对它们进行排序。

$ python3 enum_iterate.py
new = 7
incomplete = 6
invalid = 5
wont_fix = 4
in_progress = 3
fix_committed = 2
fix_released = 1

比较枚举

由于枚举成员未被排序,因此它们仅支持通过标识和相等性进行比较。

enum_comparison.py

import enum
class BugStatus(enum.Enum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
actual_state = BugStatus.wont_fix
desired_state = BugStatus.fix_released
print('Equality:',
 actual_state == desired_state,
 actual_state == BugStatus.wont_fix)
print('Identity:',
 actual_state is desired_state,
 actual_state is BugStatus.wont_fix)
print('Ordered by value:')
try:
 print('\n'.join(' ' + s.name for s in sorted(BugStatus)))
except TypeError as err:
 print(' Cannot sort: {}'.format(err))

大于和小于比较运算符引发 TypeError异常。

$ python3 enum_comparison.py
Equality: False True
Identity: False True
Ordered by value:
 Cannot sort: '<' not supported between instances of 'BugStatus
' and 'BugStatus'

将IntEnum类用于枚举,其中成员需要表现得更像数字 - 例如,以支持比较。

enum_intenum.py

import enum
class BugStatus(enum.IntEnum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
print('Ordered by value:')
print('\n'.join(' ' + s.name for s in sorted(BugStatus)))
$ python3 enum_intenum.py
Ordered by value:
 fix_released
 fix_committed
 in_progress
 wont_fix
 invalid
 incomplete
 new

唯一枚举值

具有相同值的枚举成员将作为对同一成员对象的别名引用进行跟踪。别名不会导致重复值存在于迭代器中Enum。

enum_aliases.py

import enum
class BugStatus(enum.Enum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
 by_design = 4
 closed = 1
for status in BugStatus:
 print('{:15} = {}'.format(status.name, status.value))
print('\nSame: by_design is wont_fix: ',
 BugStatus.by_design is BugStatus.wont_fix)
print('Same: closed is fix_released: ',
 BugStatus.closed is BugStatus.fix_released)

因为by_design并且closed是其他成员的别名,所以当迭代时,它们不会在输出中单独出现 Enum。成员的规范名称是附加到值的第一个名称。

$ python3 enum_aliases.py
new = 7
incomplete = 6
invalid = 5
wont_fix = 4
in_progress = 3
fix_committed = 2
fix_released = 1
Same: by_design is wont_fix: True
Same: closed is fix_released: True

要要求所有成员都具有唯一值,请将@unique 装饰器添加到Enum。

enum_unique_enforce.py

import enum
@enum.unique
class BugStatus(enum.Enum):
 new = 7
 incomplete = 6
 invalid = 5
 wont_fix = 4
 in_progress = 3
 fix_committed = 2
 fix_released = 1
 # This will trigger an error with unique applied.
 by_design = 4
 closed = 1

具有重复值的成员在解释类ValueError时会触发异常Enum。

$ python3 enum_unique_enforce.py
Traceback (most recent call last):
 File "enum_unique_enforce.py", line 11, in <module>
 class BugStatus(enum.Enum):
 File ".../lib/python3.6/enum.py", line 834, in unique
 (enumeration, alias_details))
ValueError: duplicate values found in <enum 'BugStatus'>:
by_design -> wont_fix, closed -> fix_released

以编程方式创建枚举

在某些情况下,以编程方式创建枚举更方便,而不是在类定义中对它们进行硬编码。对于这些情况,Enum还支持将成员名称和值传递给类构造函数。

enum_programmatic_create.py

import enum
BugStatus = enum.Enum(
 value='BugStatus',
 names=('fix_released fix_committed in_progress '
 'wont_fix invalid incomplete new'),
)
print('Member: {}'.format(BugStatus.new))
print('\nAll members:')
for status in BugStatus:
 print('{:15} = {}'.format(status.name, status.value))

该value参数是枚举,其被用于建立成员的表示的名称。该names参数列表枚举的成员。当传递单个字符串时,它将在空格和逗号上拆分,并且生成的标记将用作成员的名称,这些成员将自动分配以值开头的值1。

$ python3 enum_programmatic_create.py
Member: BugStatus.new
All members:
fix_released = 1
fix_committed = 2
in_progress = 3
wont_fix = 4
invalid = 5
incomplete = 6
new = 7

为了更好地控制与成员关联的值, names可以使用两部分元组序列或将名称映射到值的字典替换字符串。

enum_programmatic_mapping.py

import enum
BugStatus = enum.Enum(
 value='BugStatus',
 names=[
 ('new', 7),
 ('incomplete', 6),
 ('invalid', 5),
 ('wont_fix', 4),
 ('in_progress', 3),
 ('fix_committed', 2),
 ('fix_released', 1),
 ],
)
print('All members:')
for status in BugStatus:
 print('{:15} = {}'.format(status.name, status.value))

在此示例中,给出了由两部分组成的元组的列表,而不是仅包含成员名称的单个字符串。这使得可以BugStatus使用与定义的版本相同的顺序重建枚举的枚举enum_create.py。

$ python3 enum_programmatic_mapping.py
All members:
new = 7
incomplete = 6
invalid = 5
wont_fix = 4
in_progress = 3
fix_committed = 2
fix_released = 1

非整数成员值

枚举成员值不限于整数。实际上,任何类型的对象都可以与成员相关联。如果值是元组,则成员将作为单独的参数传递给__init__()。

enum_tuple_values.py

import enum
class BugStatus(enum.Enum):
 new = (7, ['incomplete',
 'invalid',
 'wont_fix',
 'in_progress'])
 incomplete = (6, ['new', 'wont_fix'])
 invalid = (5, ['new'])
 wont_fix = (4, ['new'])
 in_progress = (3, ['new', 'fix_committed'])
 fix_committed = (2, ['in_progress', 'fix_released'])
 fix_released = (1, ['new'])
 def __init__(self, num, transitions):
 self.num = num
 self.transitions = transitions
 def can_transition(self, new_state):
 return new_state.name in self.transitions
print('Name:', BugStatus.in_progress)
print('Value:', BugStatus.in_progress.value)
print('Custom attribute:', BugStatus.in_progress.transitions)
print('Using attribute:',
 BugStatus.in_progress.can_transition(BugStatus.new))

在此示例中,每个成员值是一个元组,其中包含数字ID(例如可能存储在数据库中)和远离当前状态的有效转换列表。

$ python3 enum_tuple_values.py
Name: BugStatus.in_progress
Value: (3, ['new', 'fix_committed'])
Custom attribute: ['new', 'fix_committed']
Using attribute: True

对于更复杂的情况,元组可能变得笨拙。由于成员值可以是任何类型的对象,因此字典可用于存在大量单独属性以跟踪每个枚举值的情况。复数值直接传递给 __init__()除了以外的唯一参数self。

enum_complex_values.py

import enum
class BugStatus(enum.Enum):
 new = {
 'num': 7,
 'transitions': [
 'incomplete',
 'invalid',
 'wont_fix',
 'in_progress',
 ],
 }
 incomplete = {
 'num': 6,
 'transitions': ['new', 'wont_fix'],
 }
 invalid = {
 'num': 5,
 'transitions': ['new'],
 }
 wont_fix = {
 'num': 4,
 'transitions': ['new'],
 }
 in_progress = {
 'num': 3,
 'transitions': ['new', 'fix_committed'],
 }
 fix_committed = {
 'num': 2,
 'transitions': ['in_progress', 'fix_released'],
 }
 fix_released = {
 'num': 1,
 'transitions': ['new'],
 }
 def __init__(self, vals):
 self.num = vals['num']
 self.transitions = vals['transitions']
 def can_transition(self, new_state):
 return new_state.name in self.transitions
print('Name:', BugStatus.in_progress)
print('Value:', BugStatus.in_progress.value)
print('Custom attribute:', BugStatus.in_progress.transitions)
print('Using attribute:',
 BugStatus.in_progress.can_transition(BugStatus.new))

此示例使用字典而不是元组表示与上一示例相同的数据。

$ python3 enum_complex_values.py
Name: BugStatus.in_progress
Value: {'num': 3, 'transitions': ['new', 'fix_committed']}
Custom attribute: ['new', 'fix_committed']
Using attribute: True

相关推荐

Java 泛型大揭秘:类型参数、通配符与最佳实践

引言在编程世界中,代码的可重用性和可维护性是至关重要的。为了实现这些目标,Java5引入了一种名为泛型(Generics)的强大功能。本文将详细介绍Java泛型的概念、优势和局限性,以及如何在...

K8s 的标签与选择器:流畅运维的秘诀

在Kubernetes的世界里,**标签(Label)和选择器(Selector)**并不是最炫酷的技术,但却是贯穿整个集群管理与运维流程的核心机制。正是它们让复杂的资源调度、查询、自动化运维变得...

哈希Hash算法:原理、应用(哈希算法 知乎)

原作者:Linux教程,原文地址:「链接」什么是哈希算法?哈希算法(HashAlgorithm),又称为散列算法或杂凑算法,是一种将任意长度的数据输入转换为固定长度输出值的数学函数。其输出结果通常被...

C#学习:基于LLM的简历评估程序(c# 简历)

前言在pocketflow的例子中看到了一个基于LLM的简历评估程序的例子,感觉还挺好玩的,为了练习一下C#,我最近使用C#重写了一个。准备不同的简历:image-20250528183949844查...

55顺位,砍41+14+3!季后赛也成得分王,难道他也是一名球星?

雷霆队最不可思议的新星:一个55号秀的疯狂逆袭!你是不是也觉得NBA最底层的55号秀,就只能当饮水机管理员?今年的55号秀阿龙·威金斯恐怕要打破你的认知了!常规赛阶段,这位二轮秀就像开了窍的天才,直接...

5分钟读懂C#字典对象(c# 字典获取值)

什么是字典对象在C#中,使用Dictionary类来管理由键值对组成的集合,这类集合被称为字典。字典最大的特点就是能够根据键来快速查找集合中的值,其键的定义不能重复,具有唯一性,相当于数组索引值,字典...

c#窗体传值(c# 跨窗体传递数据)

在WinForm编程中我们经常需要进行俩个窗体间的传值。下面我给出了两种方法,来实现传值一、在输入数据的界面中定义一个属性,供接受数据的窗体使用1、子窗体usingSystem;usingSyst...

C#入门篇章—委托(c#委托的理解)

C#委托1.委托的定义和使用委托的作用:如果要把方法作为函数来进行传递的话,就要用到委托。委托是一个类型,这个类型可以赋值一个方法的引用。C#的委托通过delegate关键字来声明。声明委托的...

C#.NET in、out、ref详解(c#.net framework)

简介在C#中,in、ref和out是用于修改方法参数传递方式的关键字,它们决定了参数是按值传递还是按引用传递,以及参数是否必须在传递前初始化。基本语义对比修饰符传递方式可读写性必须初始化调用...

C#广义表(广义表headtail)

在C#中,广义表(GeneralizedList)是一种特殊的数据结构,它是线性表的推广。广义表可以包含单个元素(称为原子),也可以包含另一个广义表(称为子表)。以下是一个简单的C#广义表示例代...

「C#.NET 拾遗补漏」04:你必须知道的反射

阅读本文大概需要3分钟。通常,反射用于动态获取对象的类型、属性和方法等信息。今天带你玩转反射,来汇总一下反射的各种常见操作,捡漏看看有没有你不知道的。获取类型的成员Type类的GetMembe...

C#启动外部程序的问题(c#怎么启动)

IT&OT的深度融合是智能制造的基石。本公众号将聚焦于PLC编程与上位机开发。除理论知识外,也会结合我们团队在开发过程中遇到的具体问题介绍一些项目经验。在使用C#开发上位机时,有时会需要启动外部的一些...

全网最狠C#面试拷问:这20道题没答出来,别说你懂.NET!

在竞争激烈的C#开发岗位求职过程中,面试是必经的一道关卡。而一场高质量的面试,不仅能筛选出真正掌握C#和.NET技术精髓的人才,也能让求职者对自身技术水平有更清晰的认知。今天,就为大家精心准备了20道...

C#匿名方法(c#匿名方法与匿名类)

C#中的匿名方法是一种没有名称只有主体的方法,它提供了一种传递代码块作为委托参数的技术。以下是关于C#匿名方法的一些重要特点和用法:特点省略参数列表:使用匿名方法可省略参数列表,这意味着匿名方法...

C# Windows窗体(.Net Framework)知识总结

Windows窗体可大致分为Form窗体和MDI窗体,Form窗体没什么好细说的,知识点总结都在思维导图里面了,下文将围绕MDI窗体来讲述。MDI(MultipleDocumentInterfac...