装饰器可以通过附加功能来增强函数,但通用装饰器通常会取代原始函数的签名带有通配符签名的签名,例如“args,*kwargs”。这可能会出现问题,尤其是在生成文档或内省函数元数据时。
装饰器模块提供了一个简单的解决方案。通过用decorator.decorator包装装饰器函数,我们可以保留原始函数的签名:
<code class="python">import decorator @decorator.decorator def args_as_ints(f, *args, **kwargs): args = [int(x) for x in args] kwargs = dict((k, int(v)) for k, v in kwargs.items()) return f(*args, **kwargs) @args_as_ints def funny_function(x, y, z=3): """Computes x*y + 2*z""" return x*y + 2*z</code>
这种方法保留了原始函数的签名:
<code class="python">help(funny_function) # Help on function funny_function in module __main__: # # funny_function(x, y, z=3) # Computes x*y + 2*z</code>
在 Python 3.4 及更高版本中,functools.wraps() 自动保留原始函数签名:
<code class="python">import functools def args_as_ints(func): @functools.wraps(func) def wrapper(*args, **kwargs): args = [int(x) for x in args] kwargs = dict((k, int(v)) for k, v in kwargs.items()) return func(*args, **kwargs) return wrapper @args_as_ints def funny_function(x, y, z=3): """Computes x*y + 2*z""" return x*y + 2*z help(funny_function) # Help on function funny_function in module __main__: # # funny_function(x, y, z=3) # Computes x*y + 2*z</code>
但是,functools.wraps() 在早期版本的 Python 中并未表现出此行为.
在函数的文档字符串中复制签名或为每个特定签名创建自定义装饰器都是有缺陷的解决方法,会引入重复和维护问题。
通过利用装饰器模块或利用 Python 3.4 中的 functools.wraps() ,可以通用地使用装饰器,同时保留原始函数签名,确保强大的文档和自省功能。
以上是使用通用装饰器时如何保留原始函数签名?的详细内容。更多信息请关注PHP中文网其他相关文章!