Каков правильный способ лечения Python argparse.Namespace() в качестве словаря?

Если я хочу использовать результаты argparse.ArgumentParser(), который является объектом Namespace, с помощью метода, который ожидает словарь или объект, подобный сопоставлению (см. collections.Mapping), каков правильный способ сделать это?

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']

Правильно ли "попасть в" объект и использовать его свойство __dict__?

Я бы подумал, что ответ отрицательный: __dict__ пахнет как соглашение для реализации, но не для интерфейса, как выглядит __getattribute__ или __setattr__ или __contains__.

Ответ 1

Вы можете получить доступ к словарю с пространством имен vars():

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

Вы можете изменить словарь напрямую, если хотите:

>>> d['baz'] = 'store me'
>>> args.baz
'store me'

Да, можно получить доступ к атрибуту __dict__. Это четкое, проверенное и гарантированное поведение.

Ответ 2

Прямо от устья лошади:

Если вы предпочитаете иметь диктофонный вид атрибутов, вы можете использовать стандартную идиому Python, vars():

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

Ответ 3

Правильно ли "попасть в" объект и использовать его свойство dict?

В общем, я бы сказал "нет". Однако Namespace поразил меня как чрезмерное, возможно, из-за того, что классы не могли наследовать от встроенных типов.

С другой стороны, Namespace представляет собой ориентированный на задачи подход к argparse, и я не могу представить ситуацию, которая потребует захвата __dict__, но пределы моего воображения не совпадают как ваш.