1. 使用枚举类
    2. 阅读: 133525
    3. 当我们需要定义常量时,一个办法是用大写变量通过整数来定义,例如月份:
    4. JAN = 1
    5. FEB = 2
    6. MAR = 3
    7. ...
    8. NOV = 11
    9. DEC = 12
    10. 好处是简单,缺点是类型是int,并且仍然是变量。
    11. 更好的方法是为这样的枚举类型定义一个class类型,然后,每个常量都是class的一个唯一实例。Python提供了Enum类来实现这个功能:
    12. from enum import Enum
    13. Month = Enum('Month', ('Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'))
    14. 这样我们就获得了Month类型的枚举类,可以直接使用Month.Jan来引用一个常量,或者枚举它的所有成员:
    15. for name, member in Month.__members__.items():
    16. print(name, '=>', member, ',', member.value)
    17. value属性则是自动赋给成员的int常量,默认从1开始计数。
    18. 如果需要更精确地控制枚举类型,可以从Enum派生出自定义类:
    19. from enum import Enum, unique
    20. @unique
    21. class Weekday(Enum):
    22. Sun = 0 # Sun的value被设定为0
    23. Mon = 1
    24. Tue = 2
    25. Wed = 3
    26. Thu = 4
    27. Fri = 5
    28. Sat = 6
    29. @unique装饰器可以帮助我们检查保证没有重复值。
    30. 访问这些枚举类型可以有若干种方法:
    31. >>> day1 = Weekday.Mon
    32. >>> print(day1)
    33. Weekday.Mon
    34. >>> print(Weekday.Tue)
    35. Weekday.Tue
    36. >>> print(Weekday['Tue'])
    37. Weekday.Tue
    38. >>> print(Weekday.Tue.value)
    39. 2
    40. >>> print(day1 == Weekday.Mon)
    41. True
    42. >>> print(day1 == Weekday.Tue)
    43. False
    44. >>> print(Weekday(1))
    45. Weekday.Mon
    46. >>> print(day1 == Weekday(1))
    47. True
    48. >>> Weekday(7)
    49. Traceback (most recent call last):
    50. ...
    51. ValueError: 7 is not a valid Weekday
    52. >>> for name, member in Weekday.__members__.items():
    53. ... print(name, '=>', member)
    54. ...
    55. Sun => Weekday.Sun
    56. Mon => Weekday.Mon
    57. Tue => Weekday.Tue
    58. Wed => Weekday.Wed
    59. Thu => Weekday.Thu
    60. Fri => Weekday.Fri
    61. Sat => Weekday.Sat
    62. 可见,既可以用成员名称引用枚举常量,又可以直接根据value的值获得枚举常量。
    63. 练习
    64. Studentgender属性改造为枚举类型,可以避免使用字符串:
    65. # -*- coding: utf-8 -*-
    66. from enum import Enum, unique
    67. class Gender(Enum):
    68. Male = 0
    69. Female = 1
    70. class Student(object):
    71. def __init__(self, name, gender):
    72. self.name = name
    73. self.gender = gender
    74. # 测试:
    75. bart = Student('Bart', Gender.Male)
    76. if bart.gender == Gender.Male:
    77. print('测试通过!')
    78. else:
    79. print('测试失败!')
    80. Run
    81. 小结
    82. Enum可以把一组相关常量定义在一个class中,且class不可变,而且成员可以直接比较。
    83. 参考源码
    84. use_enum.py