允许使用nargs的位置命令行参数通过标志符进行分隔。

9

我有一个使用argparse的程序。它需要1个必需的位置参数、1个可选的位置参数和1个标志参数。

就像这样:

usage: test.py [-h] [-a A] b [c]

所以,我尝试使用了这个:

parser = argparse.ArgumentParser()

parser.add_argument('-a')
parser.add_argument('b')
parser.add_argument('c', nargs='?', default=None)
print(parser.parse_args())

对于test.py B C -a Atest.py -a A B C都可以正常工作。

但是当我执行test.py B -a A C时,它会抛出一个错误:

$ python3 test.py B -a A C
usage: test.py [-h] [-a A] b [c]
test.py: error: unrecognized arguments: C

那么,我该如何使它接受可选位置参数,即使中间有标志呢?

请注意,如果我删除 nargs='?', default=None,那么这将起作用,但这样它就不是可选的了。同样,nargs='*' 也会出现这个问题,但是对于 nargs=N(例如 nargs=1nargs=2),以及 nargs='+' 它不会发生。nargs=argparse.REMAINDER 会将标志解析为 c 的一部分(c = ['-a', 'A', 'C']a = None)。

1个回答

9
这是一个已知的问题,无论是在SO还是Python bug/issues上都有记录,且没有简单的解决方法。 https://bugs.python.org/issue15112 这是基本解析算法的结果。它尝试解析位置参数直到下一个可选标志。然后解析带有标志的选项(以及所需的任意数量的参数)。然后继续解析下一批位置参数等。
当解析器处理b时,即使只有一个字符串,它也可以处理cc 不需要任何东西。这意味着c在第一次处理位置参数时就被“用完”了。
In [50]: parser.parse_args(['one'])
Out[50]: Namespace(a=None, b='one', c=None)
In [51]: parser.parse_args(['one','two'])
Out[51]: Namespace(a=None, b='one', c='two')
In [52]: parser.parse_args(['one','-a','1','two'])
usage: ipython3 [-h] [-a A] b [c]
ipython3: error: unrecognized arguments: two
An exception has occurred, use %tb to see the full traceback.

SystemExit: 2

/home/paul/.local/lib/python3.6/site-packages/IPython/core/interactiveshell.py:2971: UserWarning: To exit: use 'exit', 'quit', or Ctrl-D.
  warn("To exit: use 'exit', 'quit', or Ctrl-D.", stacklevel=1)
In [53]: parser.parse_known_args(['one','-a','1','two'])
Out[53]: (Namespace(a='1', b='one', c=None), ['two'])

当使用c(即使它只是获取默认值)时,没有东西可以消耗最后的字符串。它是一个“额外”的字符串。

parse_intermixed_args

Python 3.7添加了一个解决此问题的解析方法parse_intermixed_argshttps://docs.python.org/3/library/argparse.html#intermixed-parsing

In [447]: import argparse37
In [448]: p = argparse37.ArgumentParser()
In [449]: p.add_argument('pos1');
In [450]: p.add_argument('-a');
In [451]: p.add_argument('pos2', nargs='?');

In [453]: p.parse_args('1 2 -a foo'.split())
Out[453]: Namespace(a='foo', pos1='1', pos2='2')

In [454]: p.parse_args('1 -a foo 2'.split())
usage: ipython3 [-h] [-a A] pos1 [pos2]
ipython3: error: unrecognized arguments: 2
...

In [455]: p.parse_intermixed_args('1 -a foo 2'.split())
Out[455]: Namespace(a='foo', pos1='1', pos2='2')

In [456]: p.parse_intermixed_args('1 2 -a foo'.split())
Out[456]: Namespace(a='foo', pos1='1', pos2='2')

它被添加为一种允许在 '*' 位置中标记 Action 的方式。但事实上,在 '?' Actions 中也可以使用。请注意文档中的警告;它可能无法处理所有 argparse 特性。
实际上,它会禁用 positionals,并执行 parse_known_args 来获取所有 optionals,然后仅使用 positionals 解析 extras。有关详细信息,请参阅 parse_known_intermixed_args 的代码。

如果位置参数是整数有符号数字呢?(例如-1或-0.5)这是一个值,而不是另一个标志... - Kovalex
@Kovalex,通常负数会被视为普通字符串(位置参数)。可以定义一个带有标志的参数,例如“-2”。这将改变所有“负数”的处理方式。 - hpaulj
这不是nargs='?'的情况,对吧?到目前为止我找到的唯一解决方法是在选项和位置参数之间使用'--',以避免出现负数。 - Kovalex

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接