在了解裝飾器之前,我們需要知道什么閉包是什么鬼!
閉包:在一個函數內定義了一個函數f,并且這個函數f引用外部變量,在把這個函數f當做返回值返回。
上述說了閉包的三個條件:
1 函數內定義了一個函數f
2 f函數引用了外部變量
3 f被當做返回值返回
def t1():#定義t1函數 x=2 def f():#t1函數內部定義了f函數 print(x)#f函數引用了不屬于自己內部的變量x return f #f被當做返回值返回
而裝飾器有是什么鬼呢?其實閉包的一種運用。
裝飾器:在不改變函數(當然還有其他的)的代碼和調用方式的前提下,為函數增加新的功能!
那么需求來了:例如我定義了一個下載方法,我在想不改變下載方法的代碼和調用方式,為其加一個需要登錄后才能下載的功能
def get(file):#下載方法 print('下載成功',file)
那么如何實現呢?這里就要用到上面提到的閉包。
我們再來看看閉包,即是在一個函數內定義了一個函數f,并且這個函數f引用外部變量,在把這個函數f當做返回值返回
那么上面的需求我們可以先試著做第一步,即是不改變下載方法的代碼,為其加一個需要登錄后才能下載的功能
def get(file):#下載方法 print('下載成功',file)def auth(get,*args,**kwargs): print('登錄方法') get(*args,**kwargs)auth(get,'a.txt')#結果:#登錄方法#下載成功 a.txt
但是這樣的話調用方式改變了,那么我們可以利用閉包:
def get(file):#下載方法 print('下載成功',file)def func(get): def auth(): print('登錄方法') get() return authget=func(get)get('dasd')
這樣的話,參數有傳不進去,并且報錯參數沒有接收到:
我們分析一下上面代碼 func(get)
其實應該是auth這函數名,那么get=func(get),
就是把auth賦值給get變量,即是次數 get=auth,
那么 我們調用get('dasd') 即是auth('dasd'),即是我們需要修改auth函數
def get(file):#下載方法 print('下載成功',file)def func(get): def auth(file): print('登錄方法') get(file) return authget=func(get)get('dasd')
這樣的話get('dasd')的時候就是調用auth('dasd'),那么會執行登錄方法,并且執行get('dasd'),這里是真正的下載方法
優化一下就是這樣的:
def auth(f): def wrapper(*args,**kwargs): print('其它功能或方法') f(*args,**kwargs) return wrapper
這樣的要裝飾get函數的時候,只需get=auth(get),這里第一個get是變量,第二個get是函數名,這樣就實現get不改變其的代碼和調用方式,為其增加功能。
在python中 get=auth(get),可以省略為:
新聞熱點
疑難解答