Python argparse自定义操作,附带传递额外参数

28
import argparse

class customAction(argparse.Action):
    def __call__(self, parser, args, values, option_string=None):
       setattr(args, self.dest, values)

parser = argparse.ArgumentParser()
parser.add_argument('-e', '--example', action=customAction)

当触发选项 -e 时,我想将其他参数传递给 customAction,例如另一个类的实例。 我该如何做? 我尝试过的所有方法都出错了。

2个回答

51
def make_action(additional_arg):
    class customAction(argparse.Action):
        def __call__(self, parser, args, values, option_string=None):
            print(additional_arg)
            setattr(args, self.dest, values)
    return customAction
#...
parser.add_argument('-e', '--example', action=make_action('your arg'))

22
另一种解决方案是像这样派生基类argparse.Action
class CustomAction(argparse.Action):
    def __init__(self,option_strings,
                 additional_arg1,additional_arg2,
                 dest=None,
                 nargs=0,
                 default=None,
                 required=False,
                 type=None,
                 metavar=None,
                 help=None):
        self._a1=additional_arg1
        self._a2=additional_arg2
        super(CustomAction, self).__init__(
            option_strings=option_strings,
            dest=dest,
            nargs=nargs,
            default=default,
            required=required,
            metavar=metavar,
            type=type,
            help=help)
    def __call__(self, parser, namespace, values, option_string=None):
        print(self._a1)
        print(self._a2)
        setattr(namespace, self.dest, values)

#........
parser.add_argument('-e', '--example', action=CustomAction, additional_arg1='your arg', additional_arg2=42)

或者,通过提供*args**kwargs来将任何额外的参数传递给父构造函数。
class CustomAction(argparse.Action):
    def __init__(self, option_strings, additional_arg1, additional_arg2,
                 *args, **kwargs):
        self._a1 = additional_arg1
        self._a2 = additional_arg2
        super(CustomAction, self).__init__(option_strings=option_strings,
                                           *args, **kwargs)
    def __call__(self, parser, namespace, values, option_string=None):
        print(self._a1)
        print(self._a2)
        setattr(namespace, self.dest, values)

#........
parser.add_argument('-e', '--example', action=CustomAction, additional_arg1='your arg', additional_arg2=42)

尽管我喜欢闭包,但这个答案给我的印象更干净(如果更面向对象)。使用这个答案的人可以考虑添加*args**kwargs而不是列出所有argparse的默认Action参数。 - Alexander Bauer
4
这个例子中,我需要改变setattr函数的参数args为setattr函数的参数namespace,这样才能使其正常工作。 - kdubs
@kdubs 我对答案进行了编辑,纠正了那个问题。 - FeRD

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接