Python如何解除一个装饰器
问题
一个装饰器已经作用在一个函数上,你想撤销它,直接访问原始的未包装的那个函数。
解决方案
假设装饰器是通过@wraps来实现的,那么你可以通过访问__wrapped__属性来访问原始函数:
>>>@somedecorator >>>defadd(x,y): ...returnx+y ... >>>orig_add=add.__wrapped__ >>>orig_add(3,4) 7 >>>
讨论
直接访问未包装的原始函数在调试、内省和其他函数操作时是很有用的。但是我们这里的方案仅仅适用于在包装器中正确使用了@wraps或者直接设置了__wrapped__属性的情况。
如果有多个包装器,那么访问__wrapped__属性的行为是不可预知的,应该避免这样做。在Python3.3中,它会略过所有的包装层,比如,假如你有如下的代码:
fromfunctoolsimportwraps defdecorator1(func): @wraps(func) defwrapper(*args,**kwargs): print('Decorator1') returnfunc(*args,**kwargs) returnwrapper defdecorator2(func): @wraps(func) defwrapper(*args,**kwargs): print('Decorator2') returnfunc(*args,**kwargs) returnwrapper @decorator1 @decorator2 defadd(x,y): returnx+y
下面我们在Python3.3下测试:
>>>add(2,3) Decorator1 Decorator2 5 >>>add.__wrapped__(2,3) 5 >>>
下面我们在Python3.4下测试:
>>>add(2,3) Decorator1 Decorator2 5 >>>add.__wrapped__(2,3) Decorator2 5 >>>
最后要说的是,并不是所有的装饰器都使用了@wraps,因此这里的方案并不全部适用。特别的,内置的装饰器@staticmethod和@classmethod就没有遵循这个约定(它们把原始函数存储在属性__func__中)。
以上就是Python如何解除一个装饰器的详细内容,更多关于Python解除装饰器的资料请关注毛票票其它相关文章!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。