python argparse:无法识别的参数

问题描述 投票:0回答:6

当我运行

parsePlotSens.py -s bw hehe
时,它说
hehe
是一个无法识别的参数。但是,如果我运行
parsePlotSens.py hehe -s bw
,那就没问题了。理想情况下,我希望它适用于这两种情况。

有什么建议吗?以下是我的代码:

if __name__ == '__main__' :

    parser = argparse.ArgumentParser(prog='parsePlotSens');
    parser.add_argument('-s', '--sort', nargs =1, action = 'store', choices = ['mcs', 'bw'], default='mcs', help=sorthelp)
    parser.add_argument('filename', nargs ='+', action = 'store')
    option = parser.parse_args(sys.argv)
python argparse
6个回答
98
投票

请勿将

sys.argv
作为参数传递给
parse_args
。只需使用

option = parser.parse_args()

如果您确实将

sys.argv
传递给
parse_args
,则脚本本身的路径或名称是
sys.argv
中的第一项,因此成为
option.filename
的值。然后
hehe
就变成了一个未知的参数。

如果省略

sys.argv
,则
parse_args
将按预期解析
sys.argv


83
投票

您可以通过允许未知参数来解决这个问题

更换

args = parser.parse_args()

args, unknown = parser.parse_known_args()

5
投票

此外,作为对 unutbu 答案的补充,以这种方式将参数存储在字典中使测试变得容易:

args = vars(parser.parse_args())
print args

打印词典:

{'sort': ['bw'], 'filename': ['hehe']}

喜欢:

if args['sort'] == 'bw':
    # code here

...

2
投票

我的情况和问题不一样,但错误是一样的。

我的情况:

  1. 我有一个带有windows pycharm的远程开发(SFTP),并上传以在linux上运行。
  2. python 命令在我的 bash 文件中有一些带有

    \
    的换行符,例如

    python args_config.py \
        --arg1="hello" \
        --arg2="world"
    

并引发

python argparse: unrecognized arguments
args not found 错误。

问题是bash文件换行在windows和linux中是不同的,

只需用pycharm设置

File -> Line Separators -> LF - Unix and OS X (\n)

上传到linux并运行bash文件,就可以了!


2
投票

为了完成这个答案,我在这里提供了一个获取和解析未知参数的示例:

import argparse

parser = argparse.ArgumentParser()
# we only have a know argument  as key-pair --known KNOWN
parser.add_argument('--known')

# test with known un unknown variables
args, unknown = parser.parse_known_args(["--known", "var", "--unknown", "bar", "--flag"])

unknown
返回类似
["--unknown", "bar", "--flag"]
的列表。我们只需要解析它:

keypairs = dict([unknown[i:i+2] for i in range(0, len(unknown), 1) if unknown[i].startswith("--") and not (unknown[i+1:i+2]+["--"])[0].startswith("--")])

flags = [unknown[i] for i in range(0, len(unknown), 2) if (unknown[i+1:i+2]+["--"])[0].startswith("--")]

1
投票

非常有用的线程。我和@Yan Zhu有几乎相同的问题,@unutbu和@FacePalm的答案很好,但我也需要接受argv。我想出了这个,觉得很好,因为它允许我编写不需要 sys.argv 参数的单元测试。

import argparse, sys


def parse(arg_list):
     p = argparse.ArgumentParser(description="my simple app")
     p.add_argument('-z', '--zeta', type=str, default='[zeta from default]')
     p.add_argument('-a', '--application', type=str, default='[application from default]')
     return p.parse_known_args(arg_list)


code_args = [ '-a', 'a from code', '-q', 'q from code', '-o', 'o from code']

print(parse(code_args + sys.argv[1:]))

当您像这样从 intellij 添加运行时参数时

-a 'a from intellij'
结果看起来像这样。

/usr/local/bin/python3.7 /Users/me/IdeaProjects/co-util-py/test/varargstest.py -a "a from intellij"

(Namespace(application='a from intellij', other='o from code'), ['-q', 'q from code'])

您可以看到 argparse 不会删除 q,但它也不会解析它。

此外,经过大量的脑力劳动和测试,sys.argv 和创建的列表之间的唯一真正区别是

sys.argv[0]
是正在调用的程序的名称。从列表中删除它并不重要。

© www.soinside.com 2019 - 2024. All rights reserved.