为什么这个argparse代码在Python 2和3之间表现不同?

时间:2014-04-10 14:27:36

标签: python python-3.x argparse python-2.x

下面的代码,使用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'

1 个答案:

答案 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

注意 - 需要设置destrequired。需要dest才能在错误消息中为此参数指定名称。


此错误:

AttributeError: 'Namespace' object has no attribute 'do'
生成

是因为cmd子分析程序没有运行,并且没有将其参数(默认或不是)放入命名空间。您可以通过定义另一个子分析器并查看生成的args来查看该效果。