今天我就帶領(lǐng)大家由淺入深的去窺探一下碗旅,這個裝飾器到底是何方神圣岂丘,看完本篇卿樱,裝飾器就再也不是難點了.
1.什么是裝飾器
網(wǎng)上有人是這么評價裝飾器的竞阐,我覺得寫的很有趣缴饭,比喻的很形象
每個人都有的內(nèi)褲主要是用來遮羞,但是到了冬天它沒法為我們防風御寒骆莹,腫木辦颗搂?
我們想到的一個辦法就是把內(nèi)褲改造一下,讓它變得更厚更長幕垦,這樣一來丢氢,它不僅有遮羞功能傅联,還能提供保暖,不過有個問題疚察,這個內(nèi)褲被我們改造成了長褲后蒸走,雖然還有遮羞功能,但本質(zhì)上它不再是一條真正的內(nèi)褲了稍浆。于是聰明的人們發(fā)明長褲
在不影響內(nèi)褲的前提下载碌,直接把長褲套在了內(nèi)褲外面,這樣內(nèi)褲還是內(nèi)褲衅枫,有了長褲后再也不冷了
裝飾器就像我們這里說的長褲嫁艇,在不影響內(nèi)褲作用的前提下,給我們的身子提供了保暖的功效
2.裝飾器的前傳4步曲
為什么講裝飾器要放在入門篇的末尾講呢弦撩,因為這貨太難了步咪,里面有許多前提要懂。Python的函數(shù)相信大家都懂了益楼,但是大家知不知道函數(shù)也是一種對象猾漫,可以像參數(shù)一樣傳遞,我們看下面的例子吧:
1)函數(shù)也是對象
def message(word='hello'):
return word.upper()+'!'
print message()
>>
HELLO!
my_message=message
print my_message
>>
print my_message()
>>
HELLO!
也就是說message可以賦值給另外一個變量
2)函數(shù)可以嵌套感凤,定義在另外一個函數(shù)內(nèi)部
def show():
print 'Run in show()'
def message(word='hello'):
return word
print message()
show()
>>
Run in show()
hello
message可以嵌套在show函數(shù)里面悯周,調(diào)用show的時候也會跑下message函數(shù)
3)函數(shù)作為參數(shù)返回
一個函數(shù)還可以當做另外一個函數(shù)的返回值,不信看下面這個例子:
>>
hello...
4)函數(shù)作為參數(shù)傳入
我們先創(chuàng)建一個getName函數(shù),然后把這個函數(shù)當做參數(shù)傳遞給foo函數(shù)
>>
I will call the getName function later
leo
3.裝飾器的真面目
1).經(jīng)過前面幾步陪竿,大家應(yīng)該理解了函數(shù)可以作為參數(shù)傳入禽翼,也可以當參數(shù)返回,而且還可以嵌套
裝飾器其實就是對函數(shù)進行再次包裝族跛,它能夠在不改變函數(shù)的前提下闰挡,增加函數(shù)的功能,可以在函數(shù)執(zhí)行之前或者執(zhí)行之后執(zhí)行一段代碼
a_stand_alone_function()
>>
I am a stand alone function,don't you dare modify me
a_stand_alone_function_decorated=my_new_decorator(a_stand_alone_function)
a_stand_alone_function_decorated()
>>
Before the function runs
I am a stand alone function,don't you dare modify me
After the function runs
2).使用裝飾器
看這樣簡單吧礁哄,裝飾器有一個語法糖@,直接@my_new_decorator就把上面一坨代碼輕松化解了长酗,這就是Pythonic的代碼,簡潔高效
其實相當于:
another_stand_alone_function=my_new_decorator(another_stand_alone_function)
4.為什么要使用裝飾器
裝飾器提供了一些和代碼維護性和審美相關(guān)的優(yōu)點桐绒。并且作為結(jié)構(gòu)化工具夺脾,裝飾器自然地促進了代碼的封裝,這減少了冗余性并使得未來維護和擴展變得更容易茉继。
如果大家學過Django和Flask就會知道,web框架里面大量的使用裝飾器進行代碼的封裝.我們下面看一個簡單的例子:
我們有一個主題函數(shù)word()是輸出一個字符串劳翰,我們有一個函數(shù)是把字符串變粗體,另外一個是變斜體馒疹。有了裝飾器之后佳簸,我們可以非常靈活的組合,擴展函數(shù)的功能:
輸出hello
有一點要注意,就是裝飾器的順序變了生均,結(jié)果是不一樣的.
@makeitalic
@makebold
def word():
return "hello"
print word()
輸出hello<b><i>