下面的代码,使用argparse的subparsers,在Python 3上失败,但在Python 2中按预期运行。在比较文档之后,我仍然不知道为什么。
#!/usr/bin/env python
from __future__ import print_function
from argparse import ArgumentParser
def action(args):
print(args)
if __name__ == '__main__':
std = ArgumentParser(add_help=False)
std.add_argument('standard')
ap = ArgumentParser()
sp = ap.add_subparsers()
cmd = sp.add_parser('subcommand', parents=[std], description='Do subcommand')
cmd.add_argument('arg')
cmd.set_defaults(do=action)
args = ap.parse_args()
args.do(args)
Python 2.7.6的输出是:
me@computer$ python test.py
usage: test.py [-h] {subcommand} ...
test.py: error: too few arguments
在Python 3.3.5中,我得到:
me@computer$ python3 test.py
Traceback (most recent call last):
File "test.py", line 21, in <module>
args.do(args)
AttributeError: 'Namespace' object has no attribute 'do'
答案 0 :(得分:15)
最新的argparse
版本改变了它测试所需参数的方式,而subparsers则陷入了困境。它们不再是“必需的”。 http://bugs.python.org/issue9253#msg186387
当你得到test.py: error: too few arguments
时,它反对你没有给它一个'子命令'参数。在3.3.5中,它使它超过该步骤,并返回args
。
通过此更改,3.3.5应该与早期版本的行为相同:
ap = ArgumentParser()
sp = ap.add_subparsers(dest='parser') # dest needed for error message
sp.required = True # force 'required' testing
注意 - 需要设置dest
和required
。需要dest
才能在错误消息中为此参数指定名称。
此错误:
AttributeError: 'Namespace' object has no attribute 'do'
生成是因为cmd
子分析程序没有运行,并且没有将其参数(默认或不是)放入命名空间。您可以通过定义另一个子分析器并查看生成的args
来查看该效果。