python裝飾器代碼深入講解
python裝飾器就是用于擴展原函數(shù)功能的一種函數(shù),這個函數(shù)特殊的地方就是它的返回值也是一個函數(shù),使用Python裝飾器的一個好處就是:在不需要修改原函數(shù)代碼的情況下,給函數(shù)增加新的功能。
先來看個例子:
def say(): print('Nice day') say() # 這個函數(shù)的輸出為: Nice day
現(xiàn)在,我想在輸出Nice day
的前面再打印一行****************
,類似下面的效果:
****************
Nice day
一般情況下,我可以修改上面的代碼:
def say(): print('****************') print('Nice day') say()
可是,如果我忽然發(fā)現(xiàn)自己看錯了需求,這時候又要把代碼修改到原來的樣子,慶幸的是我只是在原來函數(shù)的基礎(chǔ)上增加了一行代碼,想要回到原來的狀態(tài)并不難,可如果我是修改了復(fù)雜的邏輯,代碼有一百行呢,難道我還要一步步撤銷嗎?顯然做不到,不過沒關(guān)系,肯定還有別的辦法:
def say(): print('Nice day') def outer(): # 重新定義一個新函數(shù) print('****************') # 處理新的邏輯 say() # 再調(diào)用原來的函數(shù) outer() # 現(xiàn)在的輸出為: ''' **************** Nice day '''
怎么樣,看上去已經(jīng)滿足要求了吧,不過仔細一看,就能發(fā)現(xiàn)新的問題,如果不僅僅是say()
函數(shù)需要打印****************
,新來的talk()
函數(shù)也需要呢,這時候我又要再寫一個outer()
函數(shù)嗎?這會累壞丹丹的,所以得再想個辦法:
def say(): print('Nice day') def talk(): print('I am talk') def outer(func): # 接收一個函數(shù) print('****************') # 處理新的邏輯 func() # 調(diào)用傳遞過來的函數(shù) outer(talk) # 把talk函數(shù)作為參數(shù)傳遞過去 # 輸出結(jié)果如下: ''' **************** I am talk '''
這時,不管有幾個函數(shù)需要打印****************
,我直接把函數(shù)名傳給outer()
就可以啦,是不是方便很多^-^ 但是勤勞的丹丹會止步于此嗎?肯定不會,于是又把代碼做了如下修改:
def say(): print('Nice day') def outer(func): def inner(): print('****************') func() # 相當于 say() return inner s = outer(say) # 相當于 s = inner s() # 相當于 inner()
猜猜這次的是輸出是什么~當然還是和上面一樣啦!其實這里只是把處理邏輯的部分封裝在了一個函數(shù)里面,調(diào)用outer(say)
的時候,把say
傳給outer
,獲得返回值inner
給s
,此時的s
就相當于inner
,s()
也就相當于inner()
,所以會輸出:
**************
Nice day
這就是一個最簡單的裝飾器啦,是不是很簡單~ 但是我們每次在使用的時候還需要先賦值給一個變量(這里的s),然后再經(jīng)由s調(diào)用,未免違反了丹丹“多一行代碼都是累贅”的原則,所以我們再修改一下代碼:
def outer(func): def inner(): print('****************') func() return inner @outer # 用outer裝飾say def say(): print('Nice day') say() # 調(diào)用say函數(shù)
我把outer
和say
調(diào)換了一下位置,先定義了outer
函數(shù),@outer
表示用outer
裝飾say
,這樣直接用say()
就能實現(xiàn)我想先打印一行****************
的功能了,如果不調(diào)換兩個函數(shù)的位置,是會報NameError: name 'outer' is not defined
的錯誤的噢(作用域的原因,outer
未定義),這個應(yīng)該算是復(fù)雜一點的裝飾器了吧,哈哈
這時候很多細心同學肯定就會問了,你寫的都是無參的呀,那如果我的函數(shù)有參數(shù)怎么辦呢,參數(shù)還是不固定的又該怎么辦呢?萬能的python+聰明的丹丹當然可以解決:
# 帶參數(shù)的裝飾器 def outer(func): def inner(name): func(name) return inner @outer def say(name): print('name is %s.' % (name)) say('dandan') # name is dandan.
不過這個參數(shù)個數(shù)是固定的,萬一我又突發(fā)奇想,想多傳一個hobby
或者age
怎么辦呢?
# 帶不定參數(shù)的裝飾器 def outer(func): def inner(*args, **kwargs): func(*args, **kwargs) return inner @outer def say(name, age): print('name is %s, age is %d.' % (name, age)) @outer def talk(name, age, hobby): print('name is %s, age is %d, hobby is %s.' % (name, age, hobby)) say('dandan', 18) talk('dandan', 18, 'Coding') ''' name is dandan, age is 18. name is dandan, age is 18, hobby is Coding. '''
如果我要新增的功能有很多,一個裝飾器搞不定,怎么辦呢?我可以同時使用多個裝飾器嗎?當然可以:
# 多個裝飾器 def outer(func): def inner(*args, **kwargs): print('****************') func(*args, **kwargs) return inner def outer2(func): def inner2(*args, **kwargs): print('這里有1w+新功能') func(*args, **kwargs) return inner2 @outer @outer2 def say(name, age): print('name is %s, age is %d.' % (name, age)) @outer @outer2 def talk(name, age, hobby): print('name is %s, age is %d, hobby is %s.' % (name, age, hobby)) say('dandan', 18) talk('dandan', 18, 'Coding') ''' **************** 這里有1w+新功能 name is dandan, age is 18. **************** 這里有1w+新功能 name is dandan, age is 18, hobby is Coding. '''
要注意的是,多個裝飾器的執(zhí)行順序是從第一個裝飾器開始,執(zhí)行到最后一個裝飾器,再執(zhí)行函數(shù)本身。
到此這篇關(guān)于python裝飾器代碼深入講解的文章就介紹到這了,更多相關(guān)python裝飾器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python flask服務(wù)端響應(yīng)與重定向處理各種用法小結(jié)
這篇文章主要介紹了python flask服務(wù)端響應(yīng)與重定向處理各種用法小結(jié),本文通過實例圖文相結(jié)合給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-03-03Python學習筆記之pandas索引列、過濾、分組、求和功能示例
這篇文章主要介紹了Python學習筆記之pandas索引列、過濾、分組、求和功能,結(jié)合實例形式分析了Python針對抓取保存的csv數(shù)據(jù)使用pandas進行索引列、過濾、分組、求和等操作的相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2019-06-06Python數(shù)據(jù)類型詳解(三)元祖:tuple
本文給大家介紹的是Python數(shù)據(jù)類型中的元祖(tuple),簡單的說Tuple,與列表一樣,元素也是不可變的,但與列表不同,在一個元祖可以包含不同類型的元素2016-05-05pandas的相關(guān)系數(shù)與協(xié)方差實例
今天小編就為大家分享一篇pandas的相關(guān)系數(shù)與協(xié)方差實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-12-12