【第十八天】裝飾器

7.3

1.裝飾器

裝飾器(decorator)是一種高級(jí)py語(yǔ)法仅叫,裝飾器可以對(duì)一個(gè)函數(shù)膏斤,方法或者類進(jìn)行加工
在py中,我們有多種方法對(duì)函數(shù)和類進(jìn)行加工中剩,裝飾器從操作上入手
為函數(shù)增加額外的指令忌穿,py最初沒(méi)有裝飾器這一語(yǔ)法
裝飾器在py2.5中才出現(xiàn),最初只用于函數(shù)
在py2.6以及之后的py版本中结啼,裝飾器被進(jìn)一步用于類

我們先定義兩個(gè)簡(jiǎn)單的數(shù)學(xué)函數(shù)掠剑,一個(gè)用來(lái)計(jì)算平方和
一個(gè)用來(lái)計(jì)算平方差:

#獲得平方和
def square_sum(a,b):
    return a**2 + b**2  

#獲得平方差
def square_diff(a,b):
    return a**2 - b**2
    
if __name__ == '__main__':
    print(square_sum(3,4))    #打印25
    print(square_diff(3,4))   #打印-7

在擁有了基本的數(shù)學(xué)功能之后郊愧,我們可能想為函數(shù)增加其他的功能
比如打印輸入朴译,我們可以改寫(xiě)函數(shù)來(lái)實(shí)現(xiàn)這一點(diǎn):

#裝飾:打印輸入

def square_sum(a,b):
    print('input:',a,b)
    return a**2 + b**2

def square_diff(a,b):
    print('input:',a,b)
    return a**2 - b**2
    
if __name__ == '__main__':
    print(square_sum(3,4))
    print(square_diff(3,4))

我們修改了函數(shù)定義,為函數(shù)增加了功能属铁,從代碼中可以看到
這兩個(gè)函數(shù)在功能上的拓展有很高的相似性
都是增加了print('input:',a,b)這一打印功能
我們可以改用裝飾器眠寿,定義功能拓展本身
再把裝飾器用于兩個(gè)函數(shù):

def decorator_demo(old_function):
    def new_function(a,b):
        print('input',a,b)    #額外打印操作
        return old_function(a,b)
    return new_function

@decorator_demo
def square_sum(a,b):
    return a**2 + b**2
    
@decorator_demo
def square_diff(a,b):
    return a**2 - b**2

if __name__ == '__main__':
    print(square_sum(3,4))
    print(square_diff(3,4))

裝飾器可以用def的形式定義,如上面代碼中的decorator_demo()
裝飾器接受一個(gè)可調(diào)用的對(duì)象作為輸入?yún)?shù)红选,并返回一個(gè)新的可調(diào)用對(duì)象
裝飾器新建了一個(gè)函數(shù)對(duì)象澜公,也就是上面的new_function()
在new_function()中,我們?cè)黾恿舜蛴〉墓δ?br> 并通過(guò)調(diào)用old_function(a,b)來(lái)保留原有函數(shù)的功能

定義好裝飾器后,我們就可以通過(guò)@語(yǔ)法使用了
在函數(shù)square_sum()和square_diff()定義之前
調(diào)用@decorator_demo坟乾,實(shí)際上是將square_sum()
或square_diff()傳遞給了decorator_demo()
并將decorator_demo()返回的新的函數(shù)對(duì)象賦給原來(lái)的函數(shù)名
square_sum()和square_diff()
所以當(dāng)我們調(diào)用square_sum(3,4)的時(shí)候迹辐,實(shí)際上發(fā)生的是:

square_sum = decorator_demo(square_sum)
square_sum(3,4)

我們知道,py中的變量名和對(duì)象是分離的
變量名其實(shí)是指向一個(gè)對(duì)象的引用
從本質(zhì)上甚侣,裝飾器起到的作用就是名稱綁定(name binding)
讓同一個(gè)變量名指向一個(gè)新返回的函數(shù)對(duì)象明吩,從而達(dá)到修改函數(shù)對(duì)象的目的
只不過(guò),我們很少?gòu)氐赘暮瘮?shù)對(duì)象殷费,在使用裝飾器時(shí)
我們往往會(huì)在新函數(shù)內(nèi)部調(diào)用舊的函數(shù)印荔,以便保留舊函數(shù)功能

下面看一個(gè)更有實(shí)用功能的裝飾器,我們可以利用time包來(lái)測(cè)量程序運(yùn)行的時(shí)間
把測(cè)量程序運(yùn)行時(shí)間的功能做成一個(gè)裝飾器详羡,將這個(gè)裝飾器運(yùn)用于其他函數(shù)
將顯示函數(shù)的實(shí)際運(yùn)行時(shí)間:

import time

def decorator_timer(old_function):
    def new_function(*arg,**dict_arg):
        t1 = time.time()
        result = old_function(*arg,**dict_arg)
        t2 = time.time()
        print('time:',t2 - t1)
        return result
    return new_function
@decorator_timer
def sum(a,b):
    c = a + b
    return c
if __name__ == '__main__':
    print(sum(3,4))
time: 1.1920928955078125e-06
7

在new_function()中仍律,除調(diào)用舊函數(shù)外
還前后額外調(diào)用了一次time.time(),由于time.time()返回掛鐘時(shí)間
它們的差值反映了舊函數(shù)的運(yùn)行時(shí)間实柠,此外水泉,我們通過(guò)打包參數(shù)的方法
可以在新函數(shù)和舊函數(shù)之間傳遞所有的參數(shù)

裝飾器可以實(shí)現(xiàn)代碼的可復(fù)用性,我們可以用同一個(gè)裝飾器修飾多個(gè)函數(shù)
以便實(shí)現(xiàn)相同的附加功能窒盐,比如說(shuō)草则,在建設(shè)網(wǎng)站服務(wù)器時(shí),我們能用不同函數(shù)
表示對(duì)不同HTTP請(qǐng)求的處理蟹漓,當(dāng)我們每次處理HTTP請(qǐng)求前
都想附加一個(gè)客戶驗(yàn)證功能時(shí)炕横,那么就可以定義一個(gè)統(tǒng)一的裝飾器
作用于每一個(gè)處理函數(shù),這樣葡粒,程序能重復(fù)利用份殿,可讀性也大為提高

Python中的函數(shù)的參數(shù)定義和可變參數(shù)
https://www.cnblogs.com/tqsummer/archive/2011/01/25/1944416.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市塔鳍,隨后出現(xiàn)的幾起案子伯铣,更是在濱河造成了極大的恐慌,老刑警劉巖轮纫,帶你破解...
    沈念sama閱讀 219,039評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件腔寡,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡掌唾,警方通過(guò)查閱死者的電腦和手機(jī)放前,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)糯彬,“玉大人凭语,你說(shuō)我怎么就攤上這事×冒牵” “怎么了似扔?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,417評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵吨些,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我炒辉,道長(zhǎng)豪墅,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,868評(píng)論 1 295
  • 正文 為了忘掉前任黔寇,我火速辦了婚禮偶器,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘缝裤。我一直安慰自己屏轰,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,892評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布憋飞。 她就那樣靜靜地躺著霎苗,像睡著了一般。 火紅的嫁衣襯著肌膚如雪榛做。 梳的紋絲不亂的頭發(fā)上叨粘,一...
    開(kāi)封第一講書(shū)人閱讀 51,692評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音瘤睹,去河邊找鬼。 笑死答倡,一個(gè)胖子當(dāng)著我的面吹牛轰传,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播瘪撇,決...
    沈念sama閱讀 40,416評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼获茬,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了倔既?” 一聲冷哼從身側(cè)響起恕曲,我...
    開(kāi)封第一講書(shū)人閱讀 39,326評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎渤涌,沒(méi)想到半個(gè)月后佩谣,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,782評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡实蓬,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,957評(píng)論 3 337
  • 正文 我和宋清朗相戀三年茸俭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片安皱。...
    茶點(diǎn)故事閱讀 40,102評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡调鬓,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出酌伊,到底是詐尸還是另有隱情腾窝,我是刑警寧澤,帶...
    沈念sama閱讀 35,790評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站虹脯,受9級(jí)特大地震影響驴娃,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜归形,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,442評(píng)論 3 331
  • 文/蒙蒙 一托慨、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧暇榴,春花似錦厚棵、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,996評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至奸例,卻和暖如春彬犯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背查吊。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,113評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工仙辟, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留谤草,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,332評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像悲雳,于是被迫代替她去往敵國(guó)和親惭嚣。 傳聞我的和親對(duì)象是個(gè)殘疾皇子攒磨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,044評(píng)論 2 355