我是python和装饰器的新手,并且难于编写一个装饰器,它不仅报告了args和kwargs,而且还保留了未改变的默认kwargs。python装饰器显示传递和默认kwargs
这是我到目前为止。
def document_call(fn):
def wrapper(*args, **kwargs):
print 'function %s called with positional args %s and keyword args %s' % (fn.__name__, args, kwargs)
return fn(*args, **kwargs)
return wrapper
@document_call
def square(n, trial=True, output=False):
# kwargs are a bit of nonsense to test function
if not output:
print 'no output'
if trial:
print n*n
square(6) # with this call syntax, the default kwargs are not reported
# function square called with positional args (6,) and keyword args {}
# no output
36
square(7,output=True) # only if a kwarg is changed from default is it reported
# function square called with positional args (7,) and keyword args {'output': True}
49
“问题”是这个装饰报道说,在呼叫方通过,但没有报告在广场定义中定义的默认kwargs的ARGS。报告kwargs的唯一方法是如果它们从默认值改变,即传递给方形调用。
对于我如何获得方形定义中的kwargs的任何建议也有报道?
后续检查建议,这有助于我下面的解决方案后编辑。我改变了位置参数的输出以包含他们的名字,因为我认为它使得输出更容易理解。
import inspect
def document_call(fn):
def wrapper(*args, **kwargs):
argspec = inspect.getargspec(fn)
n_postnl_args = len(argspec.args) - len(argspec.defaults)
# get kwargs passed positionally
passed = {k:v for k,v in zip(argspec.args[n_postnl_args:], args[n_postnl_args:])}
# update with kwargs
passed.update({k:v for k,v in kwargs.iteritems()})
print 'function %s called with \n positional args %s\n passed kwargs %s\n default kwargs %s' % (
fn.__name__, {k:v for k,v in zip(argspec.args, args[:n_postnl_args])},
passed,
{k:v for k,v in zip(argspec.args[n_postnl_args:], argspec.defaults) if k not in passed})
return fn(*args, **kwargs)
return wrapper
这是一个很好的学习经验。看到针对同一问题的三种不同解决方案是很好的。感谢Answerers!
@Martijn,谢谢;我已经阅读了答案的前两行,并且会消失,学习所有关于检查的内容,并且看看我想出了什么。然后会回来看看如何真正做到这一点;-) – user5799671