無懼面試,帶你搞懂python 裝飾器
寫在之前
「裝飾器」作為 Python 高級語言特性中的重要部分,是修改函數(shù)的一種超級便捷的方式,適當(dāng)使用能夠有效提高代碼的可讀性和可維護(hù)性,非常的便利靈活。
「裝飾器」本質(zhì)上就是一個函數(shù),這個函數(shù)的特點(diǎn)是可以接受其它的函數(shù)當(dāng)作它的參數(shù),并將其替換成一個新的函數(shù)(即返回給另一個函數(shù))。
可能現(xiàn)在這么看的話有點(diǎn)懵,為了深入理解「裝飾器」的原理,我們首先先要搞明白「什么是函數(shù)對象」,「什么是嵌套函數(shù)」,「什么是閉包」。關(guān)于這三個問題我在很久以前的文章中已經(jīng)寫過了,你只需要點(diǎn)擊下面的鏈接去看就好了,這也是面試中常問的知識哦:
http://www.dhdzp.com/article/158738.htm
裝飾器
搞明白上面的三個問題,其實(shí)簡單點(diǎn)來說就是告訴你:函數(shù)可以賦值給變量,函數(shù)可嵌套,函數(shù)對象可以作為另一個函數(shù)的參數(shù)。
首先我們來看一個例子,在這個例子中我們用到了前面列出來的所有知識:
def first(fun):
def second():
print('start')
fun()
print('end')
print fun.__name__
return second
def man():
print('i am a man()')
f = first(man)
f()
上述代碼的執(zhí)行結(jié)果如下所示:
start
i am a man()
end
man
上面的程序中,這個就是 first 函數(shù)接收了 man 函數(shù)作為參數(shù),并將 man 函數(shù)以一個新的函數(shù)進(jìn)行替換??吹竭@你有沒有發(fā)現(xiàn),這個和我在文章剛開始時所說的「裝飾器」的描述是一樣的。既然這樣的話,那我們就把上述的代碼改造成符合 Python 裝飾器的定義和用法的樣子,具體如下所示:
def first(func):
def second():
print('start')
func()
print('end')
print (func.__name__)
return second
@first
def man():
print('i am a man()')
man()
上面這段代碼和之前的代碼的作用一模一樣。區(qū)別在于之前的代碼直接“明目張膽”的使用 first 函數(shù)去封裝 man 函數(shù),而上面這個是用了「語法糖」來封裝 man 函數(shù)。至于什么是語法糖,不用細(xì)去追究,你就知道是類似「@first」這種形式的東西就好了。
在上述代碼中「@frist」在 man 函數(shù)的上面,表示對 man 函數(shù)使用 first 裝飾器?!窣」 是裝飾器的語法,「first」是裝飾器的名稱。
下面我們再來看一個復(fù)雜點(diǎn)的例子,用這個例子我們來更好的理解一下「裝飾器」的使用以及它作為 Python 語言高級特性被人津津樂道的部分:
def check_admin(username):
if username != 'admin':
raise Exception('This user do not have permission')
class Stack:
def __init__(self):
self.item = []
def push(self,username,item):
check_admin(username=username)
self.item.append(item)
def pop(self,username):
check_admin(username=username)
if not self.item:
raise Exception('NO elem in stack')
return self.item.pop()
上述實(shí)現(xiàn)了一個特殊的棧,特殊在多了檢查當(dāng)前用戶是否為 admin 這步判斷,如果當(dāng)前用戶不是 admin,則拋出異常。上面的代碼中將檢查當(dāng)前用戶的身份寫成了一個獨(dú)立的函數(shù) check_admin,在 push 和 pop 中只需要調(diào)用這個函數(shù)即可。這種方式增強(qiáng)了代碼的可讀性,減少了代碼冗余,希望大家在編程的時候可以具有這種意識。
下面我們來看看上述代碼用裝飾器來寫成的效果:
def check_admin(func):
def wrapper(*args, **kwargs):
if kwargs.get('username') != 'admin':
raise Exception('This user do not have permission')
return func(*args, **kwargs)
return wrapper
class Stack:
def __init__(self):
self.item = []
@check_admin
def push(self,username,item):
self.item.append(item)
@check_admin
def pop(self,username):
if not self.item:
raise Exception('NO elem in stack')
return self.item.pop()
對比一下使用「裝飾器」和不使用裝飾器的兩種寫法,乍一看,好像使用「裝飾器」以后代碼的行數(shù)更多了,但是你有沒有發(fā)現(xiàn)代碼看起來好像更容易理解了一些。在沒有裝飾器的時候,我們先看到的是 check_admin 這個函數(shù),我們得先去想這個函數(shù)是干嘛的,然后看到的才是對棧的操作;而使用裝飾器的時候,我們上來看到的就是對棧的操作語句,至于 check_admin 完全不會干擾到我們對當(dāng)前函數(shù)的理解,所以使用了裝飾器可讀性更好了一些。
就和我在之前的文章中所講的「生成器」那樣,雖然 Python 的高級語言特性好用,但也不能亂用。裝飾器的語法復(fù)雜,通過我們在上面縮寫的裝飾器就可以看出,它寫完以后是很難調(diào)試的,并且使用「裝飾器」的程序的速度會比不使用裝飾器的程序更慢,所以還是要具體場景具體看待。
以上就是無懼面試,帶你搞懂python 裝飾器的詳細(xì)內(nèi)容,更多關(guān)于python 裝飾器的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Pandas如何對帶有Multi-column(多列名稱)的數(shù)據(jù)排序并寫入Excel中
這篇文章主要介紹了Pandas如何對帶有Multi-column(多列名稱)的數(shù)據(jù)排序并寫入Excel中問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-02-02
Python通過四大 AutoEDA 工具包快速產(chǎn)出完美數(shù)據(jù)報告
在三年前,我們做數(shù)據(jù)競賽或者數(shù)據(jù)建模類的項目時,前期我們會耗費(fèi)較多的時間去分析數(shù)據(jù),但現(xiàn)在非常多擅長數(shù)據(jù)分析的大師們已經(jīng)將我們平時??吹臄?shù)據(jù)方式進(jìn)行了集成,開發(fā)了很多AutoEDA的工具包。可以幫助我們節(jié)省大量時間2021-11-11
Python shutil模塊實(shí)現(xiàn)文件的裁剪、壓縮與解壓縮的方法
這篇文章主要介紹了Python shutil模塊實(shí)現(xiàn)文件的裁剪、壓縮與解壓縮的方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-01-01
深入理解Python虛擬機(jī)中字節(jié)(bytes)的實(shí)現(xiàn)原理及源碼剖析
在本篇文章當(dāng)中主要給大家介紹在?cpython?內(nèi)部,bytes?的實(shí)現(xiàn)原理、內(nèi)存布局以及與?bytes?相關(guān)的一個比較重要的優(yōu)化點(diǎn)——?bytes?的拼接,需要的可以參考一下2023-03-03
解決pymongo連接數(shù)據(jù)庫報錯certificate verify failed:certific
這篇文章主要介紹了解決pymongo連接數(shù)據(jù)庫報錯certificate verify failed:certificate has expired問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
Python 模擬員工信息數(shù)據(jù)庫操作的實(shí)例
下面小編就為大家?guī)硪黄狿ython 模擬員工信息數(shù)據(jù)庫操作的實(shí)例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-10-10
python中List添加與刪除元素的幾種方法實(shí)例
列表基本上是?Python?中最常用的數(shù)據(jù)結(jié)構(gòu)之一了,并且刪除操作也是經(jīng)常使用的,下面這篇文章主要給大家介紹了關(guān)于python中List添加與刪除元素的相關(guān)資料,需要的朋友可以參考下2022-09-09

