本篇內(nèi)容主要講解“python裝飾器的定義及用法”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強(qiáng)。下面就讓小編來帶大家學(xué)習(xí)“python裝飾器的定義及用法”吧!
10年積累的成都網(wǎng)站制作、網(wǎng)站設(shè)計經(jīng)驗,可以快速應(yīng)對客戶對網(wǎng)站的新想法和需求。提供各種問題對應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識你,你也不認(rèn)識我。但先網(wǎng)站設(shè)計后付款的網(wǎng)站建設(shè)流程,更有內(nèi)丘免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
定義:
本質(zhì)是函數(shù)(裝飾其他函數(shù)),就是為其他函數(shù)添加附加功能。
原則:
不能修改被裝飾的函數(shù)的源代碼
不能修改被裝飾函數(shù)的調(diào)用方式
先看一個完整的裝飾器的例子:
# Author: Mr.Xue
# 2019.10.23
"""
實現(xiàn)裝飾的知識儲備:
1、函數(shù)即‘變量’
2、高階函數(shù)
3、嵌套函數(shù)
高階函數(shù)+嵌套函數(shù)--->裝飾器
"""
import time
def timmer(func):
def warpper(*args, **kwargs):
start_time = time.time()
func()
stop_time = time.time()
print('the func run time is %s' % (stop_time - start_time))
return warpper
@timmer
def test1():
time.sleep(3)
print("in the func test1")
test1()
1、函數(shù)即“變量”
函數(shù)在內(nèi)存中的存儲機(jī)制:在內(nèi)存中存儲函數(shù)體,函數(shù)名作為門牌號
注:
python中的回收機(jī)制:不管是函數(shù)還是變量,一但"門牌號"不存在了,內(nèi)存空間就會被回收
2、高階函數(shù)
符合下面兩個條件任何一個的就是高階函數(shù):
把一個函數(shù)名作為實參傳給另一個函數(shù)
返回值中包含函數(shù)名
'''
第一種高階函數(shù):
把一個函數(shù)名作為實參傳給另一個函數(shù)
--> 實現(xiàn)了不修改被裝飾的函數(shù)bar代碼的情況下為其添加新功能
'''
import time
def bar():
time.sleep(3)
print('in the bar')
def test(func):
start_time = time.time()
print(func)
func() # run function bar
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
test(bar)
#x = bar # 函數(shù)即'變量'
#x()
'''
第二種高階函數(shù):
返回值中包含函數(shù)名
--> 不修改函數(shù)的調(diào)用方式
'''
import time
def bar():
time.sleep(3)
print('in the bar')
def test2(func):
print(func) # func的內(nèi)存地址
return func #返回func的內(nèi)存地址
print(test2(bar))
bar = test2(bar)
bar()
3、嵌套函數(shù)
在一個函數(shù)的函數(shù)體中還有一個用def聲明函數(shù)
def foo():
print("in the foo")
def bar():
print("in the bar")
bar()
foo()
好啦,理解了上面的三個知識點之后,我們來一步步寫一個裝飾器,首先我們先定義兩個函數(shù)test1、test2,再給兩個函數(shù)添加新的功能,計算這兩個函數(shù)的運(yùn)行時間,怎么做呢,先定義一個高階函數(shù)(上面提到的第一種高階函數(shù)),將這兩個函數(shù)當(dāng)作參數(shù)傳進(jìn)去。
import time
def deco(func):
start_time = time.time()
func() # run function bar
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
def test1():
time.sleep(3)
print('in the test1')
def test2():
time.sleep(3)
print('in the test2')
deco(test1)
deco(test2)
現(xiàn)在我們完成了裝飾器的原則之一,再不修改源代碼的基礎(chǔ)上給函數(shù)添加了新的功能。
看上去好像不難呀,接下來,只要我們改進(jìn)代碼,完成不修改函數(shù)的調(diào)用方式的這個原則,是不是就完成了呢,好,那我們將上面的高階函數(shù)改寫成第二種的高階函數(shù)
import time
def deco(func):
start_time = time.time()
return func # 返回func函數(shù)的內(nèi)存地址
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
def test1():
time.sleep(3)
print('in the test1')
def test2():
time.sleep(3)
print('in the test2')
test1 = deco(test1)
test2 = deco(test2)
test1() #沒有修改調(diào)用方式,新功能卻不見了
test2() #沒有修改調(diào)用方式
利用了第二種高階函數(shù)之后,我們完成了裝飾器的另外一個原則,沒有改變函數(shù)的調(diào)用方式,但是發(fā)現(xiàn)新功能卻沒有加上去,這時候只用高階函數(shù)來實現(xiàn)裝飾器好像有點困難啊,怎么辦呢?好像進(jìn)入死循環(huán)了呀。
不知道大家注意到?jīng)]有,在上面那個裝飾器的例子里面,有一行寫著,“高階函數(shù)+嵌套函數(shù)—>裝飾器”,哦,那是不是我們再搞個嵌套函數(shù)就可以了呢,按照這個思路,我們繼續(xù)往下走。
import time
def timer(func):
def deco():
start_time = time.time()
func() # 調(diào)用func函數(shù)
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
return deco
def test1():
time.sleep(3)
print('in the test1')
def test2():
time.sleep(3)
print('in the test2')
test1 = timer(test1)
test2 = timer(test2)
test1() #沒有修改調(diào)用方式,新功能也加上去了
test2() #沒有修改調(diào)用方式,新功能也加上去了
哇哦,現(xiàn)在就已經(jīng)完成了裝飾器了,只不過調(diào)用方式看上去比較lower而已,只要修改成標(biāo)準(zhǔn)寫法就ok了。
怎么做呢,在要被裝飾的函數(shù)頭部,加上"@裝飾器名字",就好啦。
import time
def timer(func):
def deco():
start_time = time.time()
func() # 調(diào)用func函數(shù)
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
return deco
@timer # test1 = timer(test1)
def test1():
time.sleep(3)
print('in the test1')
@timer
def test2():
time.sleep(3)
print('in the test2')
test1()
test2()
只是現(xiàn)在我們寫的這個裝飾器和上面那個還有一點區(qū)別,我們的裝飾器沒有參數(shù),而上面的那個卻有,接下來,我們繼續(xù)完善完善。鄭州人流多少錢 http://mobile.zyyyzz.com/
給我們的內(nèi)嵌函數(shù)加上兩個非固定參數(shù),這樣就可以接收參數(shù)了,如下:
import time
def timer(func):
def deco(*args, **kwargs):
start_time = time.time()
func(*args, **kwargs) # 調(diào)用func函數(shù)
stop_time = time.time()
print("the func run time is %s" %(stop_time-start_time))
return deco
@timer # test1 = timer(test1)
def test1():
time.sleep(3)
print('in the test1')
@timer
def test2(name, age):
time.sleep(3)
print('in the test2', name, age)
test1()
test2("xue", 24)
ok, all finish!
現(xiàn)在我們用裝飾器寫一個網(wǎng)站頁面的需求:比如說,一共20個網(wǎng)站,其中除了首頁不需要驗證可以直接顯示外,其他的都需要驗證用戶名密碼才可以顯示。
# Author: Mr.Xue
# 2019.10.24
import time
user, passwd = 'xue', 'abc123'
def auth(auth_type):
print("auth_type: ", auth_type)
def outer_warpper(func):
def warpper(*args, **kwargs):
print("warpper func args:", *args, **kwargs)
if auth_type == 'local':
username = input("Username: ")
password = input("Password: ")
if user == username and passwd == password:
print("\033[32;1mUser has passed authentication\033[0m")
res = func(*args, **kwargs)
return res
else:
print("\033[31;1mInvalid username or password\033[0m")
elif auth_type == 'ldap':
print('bu hui gao mao xian')
return warpper
return outer_warpper
def index():
print("in the index")
@auth(auth_type='local') #home = warpper(home) -> warpper
def home():
print("in the home")
return 'from home'
@auth(auth_type='ldap')
def bbs():
print('in the bbs')
index()
print(home())
bbs()
到此,相信大家對“python裝飾器的定義及用法”有了更深的了解,不妨來實際操作一番吧!這里是創(chuàng)新互聯(lián)網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!