【问题标题】:Decorating a function to add custom arguments to function装饰函数以向函数添加自定义参数
【发布时间】:2013-09-14 21:53:49
【问题描述】:

我有兴趣创建一个通用函数并用它装饰所有其他函数。新函数将包含两个函数的参数。示例

def decorator(func):
    Some chunk of code

@decorator
def func(a,b):
    print a**2,b**2
    return a**2,b**2

#Equivalent func of what I want out of the decorator
def equiv_func(a,b,var):
    print var # I want to modify all decorated functions to accept a new
              # argument and print it using "print var"
    print a**2, b**2
    return a**2, b**2

这是我想出来的,但是......

def decor(func):
    def modify(var,**kwargs):
       print var
       x,y=func(**kwargs)
       return x,y
    return modify

@decor
def func(a,b):
    print a**2,b**2
    return a**2,b**2

>>> func(var="hi",a=4,b=5)
hi
16 25

我使用 **kwargs 来解释 func 的参数,但这迫使我使用 func(var="hi",a=4,b=5) 而不是 func(a=4,b= 5,var="hi") 由于 **kwargs 的定位。

此外,这种解决方法可以防止我在需要时定义装饰器时使用 **kwargs,因为它已经用于将参数输入到 func 中。示例

def test(**kwargs):
    if 'test_ans' in kwargs:
        if kwargs['test_ans']=='y':
            return True
    else:
        return False

def decor(func):
    def modify(var,**kwargs):
       if test(**kwargs): # I need to use **kwargs here, but I also need it to define my  
                          # arguments in func.
           print var
       x,y=func(**kwargs)
       return x,y
    return modify

...
# The following is what I expect but due to **kwargs being used for two purposes, 
# it doesn't work
>>>func(var='hi',a=4,b=5,test_ans='y')
'hi'
16 25
>>>func(var='hi',a=4,b=5,test_ans='n')
16 25

我希望我的问题很清楚。这些例子只是为了说明我面临的限制。它们不是实际代码(例如,我不会编写使用 **kwargs 的测试函数) 提前感谢您的任何帮助,编码大师!

编辑: 或者简单地说,有没有其他方法可以在不使用装饰器中的kwargs的情况下将func(a,b)修改为func(a,b,c,d,e,...,**kwargs)?

【问题讨论】:

  • 您可以拨打func(a=4, b=5, var="hi")
  • 首先,同意falsetru 的评论。如果您命名所有参数,您可以以任何您喜欢的方式对它们进行排序。当您开始命名事物时,定位就会被抛出窗口。唯一的要求是函数/方法定义中的所有位置参数在每次调用函数/方法时都必须有一个参数传递给它们。你也可以在定义你的装饰器时使用**kwargs...仅仅因为你在装饰器的内部函数中使用**kwargs并不意味着你不能也使用**kwargs作为装饰器本身。
  • 同意。但问题在于 **kwargs。如果我输入的可选参数多于 func 所需的参数(但我需要额外的参数来决定装饰器的行为),我会得到一个 TypeError 因为参数多于 func 需要的参数。

标签: python arguments decorator


【解决方案1】:

试试这个。您可以使用func.func_code.co_varnames 访问函数接受的参数。一旦知道这一点,您就可以过滤掉其他所有内容。

编辑:这是有限的,如果myfunc 接受**kwargs 作为参数,您将无法有效地使用它。

def print_decorator(func):
    def wrapped(*args, **kwargs):
        # a tuple of the names of the parameters that func accepts
        func_params = func.func_code.co_varnames
        # grab all of the kwargs that are not accepted by func
        extra = set(kwargs.keys()) - set(func_params)

        for kw in extra:
            print kwargs.pop(kw)

        return func(*args, **kwargs)

    return wrapped

@print_decorator
def myfunc(a, b, c=None):
    print 'myfunc was passed: a={}, b={}, c={}'.format(a, b, c)
    return a, b, c

>>> myfunc(1, b=2, c=3, d=4, e=5)
4
5
myfunc was passed: a=1, b=2, c=3

【讨论】:

  • 是的,这应该通过从 myfunc 中弹出 args 来解决 kwarg 问题。尽管您对警告是正确的,但我现在不需要它。感谢您的帮助!
【解决方案2】:

你可以允许func被调用

func(a=4, b=5, var="hi") 

通过将var 设为关键字参数。

def decor(func):
    def modify(*args, **kwargs):
        var = kwargs.pop('var', None)
        print var
        x,y=func(*args, **kwargs)
        return x,y
    return modify

@decor
def func(a,b):
    print a**2,b**2
    return a**2,b**2

func(a=4, b=5, var="hi")
func(a=4, b=5)

【讨论】:

  • 弹出 **kwargs 要求我知道弹出我可能在装饰器中使用的每个可选参数,然后再扩展 func 中剩余的 **kwargs。如果我在装饰器中有很多我需要的可选参数,这会产生很长的代码......
  • 在这种情况下,我认为您的原始代码可能是最好的。但是,如果您真的想以任意顺序调用多个关键字,则可以使用 var1, var2 = map(kwargs.pop, ('var1', 'var2')) 之类的名称。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-03-28
  • 2010-12-23
  • 2015-04-28
  • 2021-04-22
  • 2012-01-25
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多