将Python argparse.Namespace()作为字典处理的正确方法是什么?

Jas*_*n S 201 python dictionary duck-typing

如果我想使用一个需要字典或类似映射的对象的方法(参见collections.Mapping)来使用argparse.ArgumentParser()作为Namespace对象的结果,那么正确的方法是什么?

C:\>python
Python 2.7.3 (default, Apr 10 2012, 23:31:26) [MSC v.1500 32 bit (Intel)] on win
32
Type "help", "copyright", "credits" or "license" for more information.
>>> import argparse
>>> args = argparse.Namespace()
>>> args.foo = 1
>>> args.bar = [1,2,3]
>>> args.baz = 'yippee'
>>> args['baz']
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: 'Namespace' object has no attribute '__getitem__'
>>> dir(args)
['__class__', '__contains__', '__delattr__', '__dict__', '__doc__', '__eq__', '_
_format__', '__getattribute__', '__hash__', '__init__', '__module__', '__ne__',
'__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__
', '__str__', '__subclasshook__', '__weakref__', '_get_args', '_get_kwargs', 'ba
r', 'baz', 'foo']
Run Code Online (Sandbox Code Playgroud)

"伸手"进入某个物体并使用其__dict__财产是否合适?

我认为答案是否定的:__dict__闻起来像执行公约,而不是一个接口,方式__getattribute____setattr____contains__似乎是.

Ray*_*ger 320

您可以使用vars()访问命名空间的字典:

>>> import argparse
>>> args = argparse.Namespace()
>>> args.foo = 1
>>> args.bar = [1,2,3]
>>> d = vars(args)
>>> d
{'foo': 1, 'bar': [1, 2, 3]}
Run Code Online (Sandbox Code Playgroud)

如果您愿意,可以直接修改字典:

>>> d['baz'] = 'store me'
>>> args.baz
'store me'
Run Code Online (Sandbox Code Playgroud)

是的,可以访问__dict__属性.它是一种定义明确,经过测试和保证的行为.

  • @delnan有人对文档进行了不正确的编辑,并提出了过于广泛的警告.随后更正了文档.请参阅http://docs.python.org/2.7/library/functions.html#vars虽然有一些特殊情况具有只读字典(例如本地和类字典代理),但其余情况都是可更新的.*vars(obj)*调用与obj .__ dict__同义.在*argparse*命名空间的情况下,*vars(args)*可以直接访问可更新的字典. (17认同)
  • @delnan我刚刚更新了3.3和3.3文档.它将在明天可见.谢谢你指出来. (7认同)
  • @CharlieParker *dict()* 构造函数从元组列表或关键字参数构建新字典。相反,*vars()* 内置函数访问 \__dict__ 属性来检索由各种对象(包括 argparse 命名空间)封闭的现有字典。 (6认同)
  • 嗯,我想我真的不明白使用`vars()`和`__dict__`之间的区别 (2认同)

Eug*_*ash 52

直接从马的嘴里:

如果您希望使用类似于dict的属性视图,则可以使用标准的Python习惯用法vars():

>>> parser = argparse.ArgumentParser()
>>> parser.add_argument('--foo')
>>> args = parser.parse_args(['--foo', 'BAR'])
>>> vars(args)
{'foo': 'BAR'}
Run Code Online (Sandbox Code Playgroud)

- Python标准库,16.4.4.6.Namespace对象

  • @ user5359531你可能用变量覆盖了全局`vars`.您可以使用`__builtins __.vars`直接访问它,或者使用`del vars`来停止它. (5认同)