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

相關文章
相關標籤/搜索