【问题标题】:Using argparse for mandatory argument without prefix将 argparse 用于不带前缀的强制参数
【发布时间】:2022-01-18 13:53:57
【问题描述】:

我正在尝试在我的 python 应用程序中使用 argparse 模块。 我的应用程序应该使用没有任何前缀的单个强制参数运行。我想不出办法。

【问题讨论】:

  • 我假设您希望像 python myscript.py argument 这样的命令作为输入的命令。在我看来,添加一个位置参数应该可以满足您的需求。

标签: python argparse


【解决方案1】:

这是一个简单的例子,使用 argparse 只需要一个整数参数:

import argparse

parser = argparse.ArgumentParser(description='process an integer')
parser.add_argument('integer', metavar='N', type=int, nargs=1,
               help='an integer')
args = parser.parse_args()
print(args.integer)

将此代码保存在 argparse1.py 中并运行它会给出:

$ python argparse1.py 5
[5]  

$ python argparse1.py
usage: argparse1.py [-h] N
argpars1.py: error: the following arguments are required: N

$ python argparse1.py 5 7
usage: argparse1.py [-h] N
argparse1.py: error: unrecognized arguments: 7

$ python argparse1.py test
usage: argparse1.py N
argparse1.py: error: argument N: invalid int value: 'test'

$ python argparse1.py -h
usage: argparse1.py [-h] N

process an integer

positional arguments:
  N           an integer

optional arguments:
  -h, --help  show this help message and exit

要删除可选参数,请使用 add_help=False 定义 ArgumentParser,如下所示:

import argparse

parser = argparse.ArgumentParser(add_help=False)
parser.add_argument('integer', metavar='N', type=int, nargs=1)
args = parser.parse_args()
print(args.integer)

将此代码放入 argparse2.py 并对其进行测试:

$ python argparse2.py
usage: argparse2.py N
argparse2.py: error: the following arguments are required: N

$ python argparse2.py 5
[5]

$ python argparse2.py 5 7
usage: argparse2.py N
argparse2.py: error: unrecognized arguments: 7

$ python argparse2.py hello
usage: argparse2.py N
argparse2.py: error: argument N: invalid int value: 'hello'

$ python argparse2.py -h
usage: argparse2.py N
argparse2.py: error: the following arguments are required: N

可以通过导入 argparse.SUPPRESS 并将 ArgumentParser 配置为 usage=SUPPRESS 来抑制所有使用消息,如下所示:

from argparse import ArgumentParser,SUPPRESS

parser = ArgumentParser(add_help=False, usage=SUPPRESS)
parser.add_argument('integer', metavar='N', type=int, nargs=1)
args = parser.parse_args()
print(args.integer)

将此代码保存在 argparse3.py 中并运行它会给出:

$ python argparse3.py 5
[5]

$ python argparse3.py 5 7 9
argparse3.py: error: unrecognized arguments: 7 9

$ python argparse3.py -h
argparse3.py: error: the following arguments are required: N

“无法识别的参数”错误在 ArgumentParser.parse_args() (https://hg.python.org/cpython/file/3.4/Lib/argparse.py#l1727) 中进行了硬编码,并且“需要以下参数”错误单独进行了硬编码 (https://hg.python.org/cpython/file/3.4/Lib/argparse.py#l1994),两者都无需通过 API 进行修改。后者似乎在 _parse_known_args(self, arg_strings, namespace) 中,我认为可能会被覆盖,但需要复制大约 250 行代码以修改 1 行以更改该错误消息。

但是,可以通过仅添加一个带有 nargs='*' 且没有类型的参数来避免这些错误消息,这有效地允许所有命令行,然后添加自定义错误处理参数,包括您自己的错误和使用消息.例如:

import os
import sys
from argparse import ArgumentParser,SUPPRESS

script =  os.path.basename(sys.argv[0])
usage = 'usage: ' + script + ' n (int)'

parser = ArgumentParser(add_help=False, usage=SUPPRESS)
parser.add_argument('integer', nargs='*')
args = parser.parse_args()

if (len(args.integer) == 0):
    print('error: no argument provided\n', usage, file=sys.stderr, sep = '')
    sys.exit(1)


if (len(args.integer) > 1):
    print('error: too many arguments\n', usage, file=sys.stderr, sep = '')
    sys.exit(1)

v = args.integer[0]

try:
    v = int(v)
except ValueError:
    print("error: '", v, "'", ' is not an integer\n', usage, file=sys.stderr, sep='')
    sys.exit(1)

print(v + 2)

将此代码放入 argparse4.py 并对其进行测试:

$ python argparse4.py 5
7

$ python argparse4.py
error: no argument provided
usage: argparse4.py n (int)

$ python argparse4.py 5 9
error: too many arguments
usage: argparse4.py n (int)

$ python argparse4.py hello
error: 'hello' is not an integer
usage: argparse4.py n (int)

【讨论】:

  • 参数过少或过多时是否可以打印自定义错误码?
  • @Doppelganger - 错误消息格式在 hg.python.org/cpython/file/3.4/Lib/argparse.py#l1727 的 parse_args() 中硬编码,并且自定义它的能力不在 API 中。可以创建一个继承自 ArgumentParser 并覆盖该方法的新类。可以隐藏使用消息,但是我在答案的末尾添加了有关该消息的信息。
  • @Doppelganger - 可以避免硬编码错误消息并使用自定义参数处理实现您自己的错误消息。我的答案中已添加了一个示例
【解决方案2】:

您只需将参数定义为:

parser.add_argument("x")

不要在x前面添加任何-

【讨论】:

    猜你喜欢
    • 2020-07-01
    • 2013-01-24
    • 2020-09-18
    • 2014-04-15
    • 2011-08-24
    • 2016-11-08
    • 2012-08-14
    • 1970-01-01
    • 2011-12-31
    相关资源
    最近更新 更多