Python argparse - 如果没有给出参数则使用默认的互斥组

Python argparse - Mutually exclusive group with default if no argument is given

我正在编写一个 Python 脚本来处理机器可读文件并输出关于其中包含的数据的人类可读报告。
我想提供将数据输出到 stdout (-s)(默认情况下)或 txt (-t) 或 csv (-c) 文件的选项。我想要一个默认行为的开关,就像许多命令一样。

Usage: 而言,我希望看到类似 script [-s | -c | -t] input file 的内容,如果未传递任何参数,则将 -s 作为默认值。

我目前有(相关参数,简而言之):

parser = argparse.ArgumentParser()
group = parser.add_mutually_exclusive_group()
group.add_argument('-s', '--stdout', action='store_true')
group.add_argument('-c', '--csv', action='store_true')
group.add_argument('-t', '--txt', action='store_true')
args = parser.parse_args()

if not any((args.stdout, args.csv, args.txt)):
    args.stdout = True

因此,如果设置了 -s-t-c 的 none,则 stdout (-s) 将强制为 True,就好像 [=17] =] 已通过。

有没有更好的方法来实现这个?或者出于某种原因,通常会完全考虑另一种方法 'better'?

注意:我使用的是 Python 3.5.1/2 并且我不担心与其他版本的兼容性,因为此时没有计划与其他人共享此脚本。只是为了让我的生活更轻松。

您可以让每个操作都更新同一个变量,提供标准输出作为该变量的默认值。

考虑这个程序:

import argparse

parser = argparse.ArgumentParser()
group = parser.add_mutually_exclusive_group()
group.add_argument(
    '-s', '--stdout', action='store_const', dest='type', const='s', default='s')
group.add_argument(
    '-c', '--csv', action='store_const', dest='type', const='c')
group.add_argument(
    '-t', '--txt', action='store_const', dest='type', const='t')
args = parser.parse_args()
print args

您的代码可能如下所示:

if args.type == 's':
    ofile = sys.stdout
elif args.type == 'c':
    ofile = ...
...

第一个选择:

与其随意选择 .add_argument() 之一来指定默认类型,不如使用 parser.set_defaults() 来指定默认类型。

import argparse

parser = argparse.ArgumentParser()
group = parser.add_mutually_exclusive_group()
group.add_argument('-s', '--stdout', action='store_const', dest='type', const='s')
group.add_argument('-c', '--csv', action='store_const', dest='type', const='c')
group.add_argument('-t', '--txt', action='store_const', dest='type', const='t')
parser.set_defaults(type='s')
args = parser.parse_args()
print args

第二个选择:

与其将类型指定为枚举值,不如将可调用对象存储到该类型中,然后调用该可调用对象:

import argparse

def do_stdout():
    # do everything that is required to support stdout
    print("stdout!")
    return
def do_csv():
    # do everything that is required to support CSV file
    print("csv!")
    return
def do_text():
    # do everything that is required to support TXT file
    print("text!")
    return

parser = argparse.ArgumentParser()
group = parser.add_mutually_exclusive_group()
group.add_argument('-s', '--stdout', action='store_const', dest='type', const=do_stdout)
group.add_argument('-c', '--csv', action='store_const', dest='type', const=do_csv)
group.add_argument('-t', '--txt', action='store_const', dest='type', const=do_text)
parser.set_defaults(type=do_stdout)
args = parser.parse_args()
print args
args.type()

您可以 "cheat" 与 sys.argv :

import sys


def main():
    if len(sys.argv) == 2 and sys.argv[1] not in ['-s', '-c', '-t', '-h']:
        filename = sys.argv[1]
        print "mode : stdout", filename
    else:
        parser = argparse.ArgumentParser()
        group = parser.add_mutually_exclusive_group()
        group.add_argument('-s', '--stdout')
        group.add_argument('-c', '--csv')
        group.add_argument('-t', '--txt')
        args = parser.parse_args()
        if args.stdout:
            print "mode stdout :",  args.stdout
        if args.csv:
            print "mode csv :",  args.csv
        if args.txt:
            print "mode txt :",  args.txt

if __name__ == "__main__":
    main()