Python:枚举值与 nameduples

Python: enumerated values vs. nameduples

最近我参加了一个关于. Ethan Furman的讨论,建议没有必要重新发明轮子,当有标准的enum包时——没有需要玩玩。

所以,在最终切换到 3.5 之后,我尝试了 - 事实证明 enum.Enum 定义的字符串常量确实需要 取消引用,至少在某些 API 中是这样。

这是一个对我不起作用的简化示例

In [83]: consts = collections.namedtuple('consts', 'PATH')('/usr/bin')

In [84]: consts.PATH
Out[84]: '/usr/bin'

In [85]: os.path.exists(consts.PATH)
Out[85]: True

In [86]: consts = enum.Enum('consts', [['PATH','/usr/bin']])

In [87]: os.path.exists(consts.PATH)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-87-c4641f1a4c50> in <module>()
----> 1 os.path.exists(consts.PATH)

/usr/local/Cellar/python3/3.5.2_1/Frameworks/Python.framework/Versions/3.5/lib/python3.5/genericpath.py in exists(path)
     17     """Test whether a path exists.  Returns False for broken symbolic links"""
     18     try:
---> 19         os.stat(path)
     20     except OSError:
     21         return False

TypeError: argument should be string, bytes or integer, not consts

这当然有效

In [90]: os.path.exists(consts.PATH.value)
Out[90]: True

在我的书中,任何需要在整个项目中添加代码的方法都是低效的,所以 - 感谢您的建议! - 我要回到好的 ol' namedtuple 方法。

问题是 - 为什么 enum 属性不支持属性的隐式取消引用?它是 3.5.2 中的实现错误吗?

谢谢, 马克

enum.Enum 是一个基础 class。如果您在设置 class 时还继承自 str,它应该会按预期工作:

In [34]: class consts(str, enum.Enum):
   ....:     PATH = r"C:\Windows"
   ....:

In [35]: consts.PATH
Out[35]: <consts.PATH: 'C:\Windows'>

In [36]: os.path.exists(consts.PATH)
Out[36]: True