裝飾器
-
一個(gè)產(chǎn)品由多個(gè)部門(mén)協(xié)作開(kāi)發(fā),如分為底層開(kāi)發(fā)和業(yè)務(wù)開(kāi)發(fā)
- 底層開(kāi)發(fā):主要負(fù)責(zé)系統(tǒng)底層的調(diào)用,方法大概100來(lái)個(gè)
- 業(yè)務(wù)開(kāi)發(fā):調(diào)用底層開(kāi)發(fā)部門(mén)提供的方法,擴(kuò)展出上千個(gè)方法跑業(yè)務(wù)邏輯
-
如果某天老板要求需要在底層的方法都加上某個(gè)功能,可能需要把100來(lái)個(gè)方法都做修改,耗時(shí)10天
- 但是改完后老板發(fā)現(xiàn)這樣不好,需要還原,這樣就GG了,又要改回來(lái),說(shuō)不定還會(huì)把代碼改亂
- 一個(gè)優(yōu)化的方法是,在100來(lái)個(gè)方法內(nèi)調(diào)用一個(gè)外部函數(shù),在外部函數(shù)里面添加邏輯,但是這樣做有一個(gè)弊端,當(dāng)項(xiàng)目不斷交接,不同的開(kāi)發(fā)人員對(duì)底層核心代碼操作,到最后,代碼會(huì)越來(lái)越亂,所以可以用裝飾器對(duì)代碼進(jìn)行修改,不需要修改底層代碼,又不需要業(yè)務(wù)開(kāi)發(fā)人員修改調(diào)用方式
裝飾設(shè)計(jì)模式,但是調(diào)用的是裝飾后的函數(shù),python的裝飾器會(huì)覆蓋原來(lái)的函數(shù),調(diào)用的時(shí)候,函數(shù)名不變,大大減少了代碼的修改,提高擴(kuò)展性
def f1():
print('這個(gè)是原始的函數(shù)')
def f2(fun):
print('-----------function2----------start')
fun()
print('-----------function2-----------end')
f2(f1)
# 定義裝飾器
def outer(fun):
def inner(): # 定義函數(shù)inner
print('inner start')
fun()
print('inner end')
return inner # 返回函數(shù)inner
# 使用裝飾器
@outer
def fun1():
print('HelloWorld')
# 原函數(shù)調(diào)用不變
fun1()
====== >>
inner start
HelloWorld
inner end
-
原理:
- 執(zhí)行outer函數(shù),并將原函數(shù)作為參數(shù)傳入outer函數(shù)
- 將outer函數(shù)的返回值重新賦值給原函數(shù),并覆蓋原函數(shù)
- 如果返回一個(gè)inner函數(shù),則原函數(shù)名不變,函數(shù)體為inner函數(shù)的函數(shù)體
- 如果返回的是一個(gè)值,則原函數(shù)名指向的是一個(gè)值,變成一個(gè)變量
- 類(lèi)似:將原函數(shù)作為參數(shù)封裝在另一個(gè)函數(shù)內(nèi),形成新的函數(shù)
- 只要函數(shù)應(yīng)用的裝飾器,原函數(shù)就被重新定義,重定義為裝飾器內(nèi)的函數(shù)
傳參:如果原函數(shù)有參數(shù),則裝飾器返回的參數(shù)需要和原函數(shù)一樣,這樣才能保證原調(diào)用者不需要修改代碼,同時(shí),在裝飾器內(nèi),原函數(shù)能夠正常調(diào)用。
def outer(fun):
def inner(string):
print('inner')
fun(string)
print('inner')
return inner
@outer
def fun(string):
print('fun', string)
fun('string')
- 如果多個(gè)函數(shù)需要相同的操作,幾十個(gè)底層函數(shù)需要同的功能,如果每一個(gè)都要寫(xiě)一個(gè)裝飾器,這樣代碼量太大,可以考慮裝飾器復(fù)用,但是未必所有函數(shù)的參數(shù)需要的類(lèi)型和個(gè)數(shù)相同,這就需要林另外加判斷了,比較麻煩
- 但是python內(nèi)部已經(jīng)幫我們做好了優(yōu)化
- fun(*args, **kwargs)作為內(nèi)部調(diào)用函數(shù)的參數(shù)
def outer(fun):
def inner(*args, **kwargs):
print('inner')
fun(*args, **kwargs)
print('inner')
return inner
@outer
def fun1(num1, num2):
print(num1 + num2)
fun1(100, 200)
- 一個(gè)函數(shù)調(diào)用多個(gè)裝飾器
- 一個(gè)函數(shù)可以有多個(gè)裝飾器
- 應(yīng)用:某東頁(yè)面查看訂單,需要先登陸,很多地方都需要先登陸才能查看,所以通過(guò)裝飾器在查看詳情前登陸
- 登陸后,用戶有普通用戶,白金,vip等不同等級(jí)的用戶,再此基礎(chǔ)上在疊加裝飾器
- 執(zhí)行循序是從下網(wǎng)上:outer3-->outer1-->outer2
def outer2(fun):
def inner(*args, **kwargs):
print('裝飾器嵌套')
fun(*args, **kwargs)
return inner
def outer1(fun):
def inner(*args, **kwargs):
print('inner')
fun(*args, **kwargs)
print('inner')
return inner
def outer3(fun):
def inner(*args, **kwargs):
fun(*args, **kwargs)
print('outer3')
return inner
@outer2
@outer1
@outer3
def fun(string):
print('fun', string)
fun('string')