argparse中的可选nargs变量

时间:2016-06-29 20:34:59

标签: python argparse

如何使用带有变量nargs的argparse进行条件循环?所以,基本上,它应该在有或没有参数的情况下运行。 我在尝试:

parser = argparse.ArgumentParser(description="output parser")
group = parser.add_mutually_exclusive_group()
group.add_argument("--dos", help="get DOSCAR for plot",
                   nargs="?", metavar=("int"))
args = parser.parse_args()

if args.dos:    
    if len(args.dos) > 1:
        chosen = int(args.dos[0])
        chdos = "at_dos"+args.dos[0]+".dat"
    else:
        chosen = None

    inpt = "DOSY"
    print(chosen)
    print(inpt)

现在,如果我有变量,那么它打印一些值,错误但有些值:

$python3 vasp.py --dos 111
111
None   # IT SHOULDN'T BE NONE
DOSY

但没有争论。

我也尝试过普通的sys.argv,如:

def get_dos():
    if len(sys.argv) > 2:
        chosen = int(sys.argv[2])
        chdos = "at_dos"+sys.argv[2]+".dat"
    else:
        chosen = None
    inpt = "DOSCAR"
    print(sys.argv)

    print(args.dos)
    print(chosen)
    print(inpt)

在这种情况下,当选项存在时,它给出正确的结果:

python3 vasp.py --dos 12
['vasp.py', '--dos', '12']
12
12
DOSCAR

但是再一次,没有选择:

$python3 vasp.py --dos

我试过了hpaulj的建议。它给出了:

$python3 tt.py --dos 12
Namespace(dos='12')
1
DOSY

并且没有争论,它仍然没有打印任何东西。

3 个答案:

答案 0 :(得分:3)

在Ipython会话中简化您的解析器:

In [1004]: parser=argparse.ArgumentParser()    
In [1005]: parser.add_argument('--dos', nargs='?')

In [1007]: parser.parse_args('--dos 111'.split())
Out[1007]: Namespace(dos='111')

在这种情况下,args.dos将是字符串' 111',len 3和int(args.dos[0])数字1.同样的情况发生在你nargs 1}}空白(默认无)。

使用nargs='?'我也可以使用不带参数的标志,在这种情况下,该值是默认值None

In [1013]: parser.parse_args('--dos'.split())
Out[1013]: Namespace(dos=None)

nargs=is most useful with a const , which gives a convenient 3 way action. I can add type = int`将字符串转换为整数。

In [1015]: parser.add_argument('--dos', nargs='?', type=int,
   default=None, const=123)

In [1016]: parser.parse_args([]) # not used
Out[1016]: Namespace(dos=None)

In [1017]: parser.parse_args('--dos'.split())  # without argument
Out[1017]: Namespace(dos=123)

In [1018]: parser.parse_args('--dos 456'.split())  # with argument
Out[1018]: Namespace(dos=456)

其他nargs,例如1,' *'和' +'给你一个列表,你可以检查长度等。

====================

argv测试中

if len(sys.argv) > 2:
    chosen = int(sys.argv[2])
    chdos = "at_dos"+sys.argv[2]+".dat"

sys.argv是一个列表,因此如果有足够的元素来应用sys.argv[2]步骤,len会计算。

这不起作用,因为args.dos是单个字符串,而不是列表。

if len(args.dos) > 1:
    chosen = int(args.dos[0])
    chdos = "at_dos"+args.dos[0]+".dat"

len(args.dos)则是字符串中的字符数,args.dos[0]是第一个字符。

============

如果我定义:

def get_dos(argv=None):
    parser=argparse.ArgumentParser()
    parser.add_argument('--dos', type=int, nargs='?')
    args = parser.parse_args(argv)
    chosen = args.dos
    if chosen is not None:
        chdos = 'at_dos%s.dat'%chosen
    else:
        chdos = ''
    return chosen, chdos

这些测试产生了我认为符合您需求的值:

In [1042]: get_dos([])
Out[1042]: (None, '')

In [1043]: get_dos(['--dos'])
Out[1043]: (None, '')

In [1044]: get_dos(['--dos','123'])
Out[1044]: (123, 'at_dos123.dat')

答案 1 :(得分:0)

你需要替换nargs ="?"与nargs =" *"。将nargs设置为"?"意味着将有1个可选参数。你想要的是一个可选参数列表,它是" *"。像这样。

import argparse
parser = argparse.ArgumentParser(description="output parser")
group = parser.add_mutually_exclusive_group()
group.add_argument("--dos", help="get DOSCAR for plot",
               nargs="*", metavar=("int"))
args = parser.parse_args()

if args.dos:    
    if len(args.dos) > 1:
        chosen = int(args.dos[0])
        chdos = "at_dos"+args.dos[0]+".dat"
    else:
        chosen = None

    inpt = "DOSY"
    print(chosen)
    print(inpt)

答案 2 :(得分:0)

感谢所有回复的人,特别是@hpaulj。但他们都没有真正解决我的问题。所以,我采取了其他方式。我在这里发布这个是为了完整。

#!/usr/bin/python3
import argparse

parser = argparse.ArgumentParser(description="output parser")
group = parser.add_mutually_exclusive_group()
group.add_argument("--dos", help="get DOSCAR for plot",action='store_true')
parser.add_argument("-n", help="Showing last n line",
                    metavar='integer', type=int)

args = parser.parse_args()
if args.dos:
    if args.n:
        chosen = int(args.n)
        chdos = "at_dos"+str(args.n)+".dat"
    else:
        chosen = None
    inpt = "DOSY"
    print(inpt)
    print(chosen)

正确产生预期结果:

$python3 tt.py --dos
DOSY
None

$python3 tt.py --dos -n 222
DOSY
222