|
本文實例講述了Python裝飾器decorator用法。分享給大家供大家參考。具體分析如下: 1. 閉包(closure) 閉包是Python所支持的一種特性,它讓在非global scope定義的函數(shù)可以引用其外圍空間中的變量,這些外圍空間中被引用的變量叫做這個函數(shù)的環(huán)境變量。環(huán)境變量和這個非全局函數(shù)一起構(gòu)成了閉包。 復(fù)制代碼 代碼如下: def outer(x): y = [1,2,3] def inner(): print x print y return inner x = 5 #這個x沒有被引用 f = outer(2) f() print f.__closure__ #函數(shù)屬性__closure__存儲了函數(shù)的環(huán)境變量 def entrance(func): = 5 #這個x沒有被引用f = outer(2)f()print f.__closure__ #函數(shù)屬性__closure__存儲了函數(shù)的環(huán)境變量 def entrance(func): x和y都是屬于函數(shù)outer命名空間的,在inner中被引用,當(dāng)outer函數(shù)退出后,outer的命名空間不存在了,但是inner依然維護(hù)了其定義時候?qū)ζ渫獠孔兞縳,y的連接。 程序輸出: 2 [1, 2, 3] (, ) 裝飾器是一個可調(diào)用對象(a callable),在Python中,函數(shù)是對象,當(dāng)然也是可調(diào)用的,所以裝飾器可以是一個函數(shù),我們稱其為函數(shù)裝飾器。 復(fù)制代碼 代碼如下: def entrance(func): def inner(): print 'inside function :', func.__name__ func() return inner entrance是一個裝飾器,它是一個函數(shù),它可以接收一個函數(shù)func作為參數(shù),返回了另一個函數(shù)inner。 那為什么叫裝飾器了,在返回函數(shù)inner()的內(nèi)部,調(diào)用了func(),而且還作了額外的操作,相當(dāng)于“裝飾”了函數(shù)func。 那如何使用裝飾器? 復(fù)制代碼 代碼如下: def fun1(): pass fun1 = entrance(fun1) def fun2(): pass fun2 = entrance(fun2) fun1,fun2的名字都沒有變,但是通過調(diào)用函數(shù)裝飾器entrance(),它們已經(jīng)指向了另一個函數(shù)inner(),“裝飾了”自己。 @操作符 Python提供的@符號,實質(zhì)上就是上面做的,對一個函數(shù)名進(jìn)行從新賦值,是語法上的技巧。所以上面的代碼等價于 復(fù)制代碼 代碼如下: @entrance def fun1(): pass @entrance def fun2(): pass 2. 裝飾器的用途 從這個刻意構(gòu)造的很簡單的例子,可以看出裝飾器的意義,如果一個函數(shù)需要一個功能,如果這個功能可以被使用在很多函數(shù)上,或是函數(shù)并不是自己實現(xiàn),那可以寫個裝飾器來實現(xiàn)這些功能。 復(fù)制代碼 代碼如下: @entrance def fun3(x): pass 只要不調(diào)用fun3,這三行代碼是不會讓Python解釋器報錯的,因為我們已經(jīng)知道,它等價于: 復(fù)制代碼 代碼如下: def fun3(x): pass fun3 = entrance(fun3) 我們定義了一個帶參的函數(shù)fun3,然后把fun3指向了另一個函數(shù)inner(),當(dāng)然不會有什么錯。 但是,當(dāng)我們使用fun3時,我們肯定會按照它定義時的樣子去使用它,給它傳入一個參數(shù)。 >>>fun3(1) 這里就會出錯了,看看解釋器怎么報錯的 Traceback (most recent call last): 當(dāng)然我們已經(jīng)很容易知到為什么會這樣報錯了,fun3已經(jīng)不是指向它定義時那個函數(shù)了,它現(xiàn)在指向了”inner()”,而inner是沒有參數(shù)的,當(dāng)然會出錯。 3. 寫個裝飾器logger 一個函數(shù)被調(diào)用時,在日志里記錄其名稱和被調(diào)用的實際參數(shù) 希望本文所述對大家的Python程序設(shè)計有所幫助。 |
|
|