允许将带有nargs的位置命令行参数用标志分隔

发布于 2021-01-29 17:25:19

我有一个使用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 A和适用test.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使得它解析标志作为其一部分cc = ['-a', 'A', 'C']a = None

关注者
0
被浏览
178
1 个回答
  • 面试哥
    面试哥 2021-01-29
    为面试而生,有面试问题,就找面试哥。

    这是一个已知的问题,无论是关于SO还是Python错误/问题,都没有简单的解决方法。
    https://bugs.python.org/issue15112

    这是基本解析算法的结果。这将尝试解析位置信息,直到下一个可选标志。然后解析标记的选项(以及它需要的许多参数)。然后解析下一批位置,等等。

    当解析器处理时bc即使只有一个字符串,它也可以处理。 c不需要什么。这意味着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_args
    https://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')
    

    添加它是为了允许在“ *”位置中间标记操作。但是最终在这种情况下以“?”开始工作 动作。请注意文档中的注意事项;它可能无法处理所有argparse功能。

    实际上,它会停用positionals,执行aparse_known_args来获取全部optionals,然后extras仅将解析为positonals。有关parse_known_intermixed_args详细信息,请参见的代码。



知识点
面圈网VIP题库

面圈网VIP题库全新上线,海量真题题库资源。 90大类考试,超10万份考试真题开放下载啦

去下载看看