python装饰器中functools.wraps的做用详解

直接上代码看效果:html

  # 定义一个最简单的装饰器   def user_login_data(f):     def wrapper(*args, **kwargs):       return f(*args, **kwargs)     return wrapper   # 用装饰器装饰如下两个函数      @user_login_data   def num1():     print("aaa")   @user_login_data   def num2():     print("bbbb")   if __name__ == '__main__':     print(num1.__name__)     print(num2.__name__)

 

 以上代码的输出结果为:flask

    wrapper     wrapper

 

由此函数使用装饰器时,函数的函数名即 __name__已经被装饰器改变.app

通常定义装饰器的话能够不用考虑这点,可是若是多个函数被两个装饰器装饰时就报错,由于两个函数名同样,第二个函数再去装饰的话就报错.框架

解决方案就是引入  functools.wraps  ,以上代码的解决以下: 函数

         def user_login_data(f):     @functools.wraps(f)       def wrapper(*args, **kwargs):         return f(*args, **kwargs)       return wrapper

 

增长@functools.wraps(f), 能够保持当前装饰器去装饰的函数的 __name__ 的值不变测试

以上输出结果就是: url

    num1     num2

Python装饰器(decorator)在实现的时候,有一些细节须要被注意。例如,被装饰后的函数其实已是另一个函数了(函数名等函数属性会发生改变)。这样有时候会对程序形成一些不便,例如笔者想对flask框架中的一些函数添加自定义的decorator,添加后因为函数名和函数的doc发生了改变,对测试结果有一些影响。spa

因此,Python的functools包中提供了一个叫wraps的decorator来消除这样的反作用。写一个decorator的时候,最好在实现以前加上functools的wrap,它能保留原有函数的名称和docstring。.net

相关文章
相关标签/搜索