直接上代碼看效果: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