你答对了。但也许我可以解释一些细节。
a = parser.add_argument(...)
add_argument
创建一个Action
对象(或实际上是一个子类,具体取决于action
参数)。您可以在自己的环境中保存指向该对象的指针。但是该 Action 也收集在parse._actions
列表中。这就是他们parser
跟踪其论点的方式。
阅读_actions
应该始终是安全的。修改它可能会破坏解析器。 argument_groups
有权访问该列表。
subparsers = parser.add_subparsers(dest="command")
是add_argument
, 创建和返回argparse._SubParsersAction
对象的特殊版本。 subparsers
是那个物体。正如前面的答案所指出的,您可以_actions
通过搜索正确的子类在列表中找到它。(对于主解析器来说,subparsers
这只是另一个位置参数。)
subparsers
维护自己的专用字典parsers
,可作为其choices
属性访问。主解析器没有这些子解析器的任何记录。
parser_other = subparsers.add_parser("other")
创建一个解析器,将其放入该choices
映射中,并返回一个参考供您自己使用(与add_argument
等)。每个子解析器都有自己的_actions
列表。(和它自己的_defaults
)。
查看get_defaults
方法的代码:
def get_default(self, dest):
for action in self._actions:
if action.dest == dest and action.default is not None:
return action.default
return self._defaults.get(dest, None)
它使用_actions
属性。并查看action.default
Action的属性。
self._defaults
是由该parser.set_defaults
方法更新的字典。该方法还将其参数复制到相关的 Action 对象。 get_defaults
检查是否dest
是与特定操作无关的默认值之一。 https://docs.python.org/3/library/argparse.html#argparse.ArgumentParser.set_defaults
我没有太多使用该parser._subparsers
属性。看看parser.add_subparsers
我看到的方法,它实际上是一个argument_group
. Argument_groups 主要是一个help
工具,用于对帮助行进行分组。解析器对象和它的 argument_groups 之间的关系有点棘手,而且可能不是您想要使用的东西。
这是一个示例,包含更多(太多)细节:
In [22]: parser = argparse.ArgumentParser()
In [23]: sp = parser.add_subparsers(title='subparsers', dest='cmd')
In [24]: sp1 = sp.add_parser('cmd1')
In [25]: sp2 = sp.add_parser('cmd2')
In [26]: parser.print_help()
usage: ipython3 [-h] {cmd1,cmd2} ...
optional arguments:
-h, --help show this help message and exit
subparsers:
{cmd1,cmd2}
In [28]: [a.dest for a in parser._actions]
Out[28]: ['help', 'cmd']
In [29]: parser._action_groups
Out[29]:
[<argparse._ArgumentGroup at 0xaf86bf2c>,
<argparse._ArgumentGroup at 0xaf86bdcc>,
<argparse._ArgumentGroup at 0xac99fa6c>]
In [30]: [g.title for g in parser._action_groups]
Out[30]: ['positional arguments', 'optional arguments', 'subparsers']
In [31]: parser._subparsers
Out[31]: <argparse._ArgumentGroup at 0xac99fa6c>
_defaults
of_subparsers
实际上是同一个字典parser._defaults
In [32]: parser.set_defaults(extra='foobar')
In [33]: parser._defaults
Out[33]: {'extra': 'foobar'}
In [34]: parser._subparsers._defaults
Out[34]: {'extra': 'foobar'}
parser._subparsers._actions
也与 相同parser._actions
。但是该组确实维护自己的列表操作(用于帮助显示)。
In [35]: parser._subparsers._group_actions
Out[35]: [_SubParsersAction(option_strings=[], dest='cmd', nargs='A...', const=None,
default=None, type=None, choices=OrderedDict([...]), help=None, metavar=None)]
因此,您可以使用parser._subparsers._group_actions[0]
查找subparsers
操作对象而不是搜索parsers._actions
列表。
In [37]: parser._subparsers._group_actions[0].choices
Out[37]:
OrderedDict([('cmd1',
ArgumentParser(prog='ipython3 cmd1', usage=None, description=None,...)),
('cmd2',
ArgumentParser(prog='ipython3 cmd2', usage=None, description=None,...))])
再想一想,parser._subparsers._group_actions
可能没那么有用。如果你不给它一个特殊的标题,那么它与parser._positionals
所有位置参数的参数组相同。所以你仍然需要验证_SubParsersAction
类。