Python裝飾器decorator用法執行個體

來源:互聯網
上載者:User
本文執行個體講述了Python裝飾器decorator用法。分享給大家供大家參考。具體分析如下:

1. 閉包(closure)

閉包是Python所支援的一種特性,它讓在非global scope定義的函數可以引用其外圍空間中的變數,這些外圍空間中被引用的變數叫做這個函數的環境變數。環境變數和這個非全域函數一起構成了閉包。
複製代碼 代碼如下:

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__ #函數屬性__closure__儲存了函數的環境變數 def entrance(func):
= 5 #這個x沒有被引用f = outer(2)f()print f.__closure__ #函數屬性__closure__儲存了函數的環境變數 def entrance(func):
x和y都是屬於函數outer命名空間的,在inner中被引用,當outer函數退出後,outer的命名空間不存在了,但是inner依然維護了其定義時候對其外部變數x,y的串連。
程式輸出:
  2
  [1, 2, 3]
  (, )

裝飾器是一個可調用對象(a callable),在Python中,函數是對象,當然也是可調用的,所以裝飾器可以是一個函數,我們稱其為函數裝飾器。
這個可調用對象以一個函數作為參數,閉且返回另一個函數(來替換參數那個函數)。
比如:

複製代碼 代碼如下:

def entrance(func):
def inner():
print "inside function :", func.__name__
func()
return inner



entrance是一個裝飾器,它是一個函數,它可以接收一個函數func作為參數,返回了另一個函數inner。
那為什麼叫裝飾器了,在返回函數inner()的內部,調用了func(),而且還作了額外的操作,相當於“裝飾”了函數func。
那如何使用裝飾器?

複製代碼 代碼如下:

def fun1():
pass
fun1 = entrance(fun1)
def fun2():
pass
fun2 = entrance(fun2)



fun1,fun2的名字都沒有變,但是通過調用函數裝飾器entrance(),它們已經指向了另一個函數inner(),“裝飾了”自己。

@操作符

Python提供的@符號,實質上就是上面做的,對一個函數名進行從新賦值,是文法上的技巧。所以上面的代碼等價於
複製代碼 代碼如下:

@entrance
def fun1():
pass
@entrance
def fun2():
pass



2. 裝飾器的用途

從這個刻意構造的很簡單的例子,可以看出裝飾器的意義,如果一個函數需要一個功能,如果這個功能可以被使用在很多函數上,或是函數並不是自己實現,那可以寫個裝飾器來實現這些功能。
上面的裝飾器entrance,裝飾一個函數後,函數被調用時會列印出這個函數的名字。
但是有一個問題,這個裝飾器從功能上看,是要應該可以用來裝飾任何函數,但是如果我們用它來裝飾了一個帶參數的函數
複製代碼 代碼如下:

@entrance
def fun3(x):
pass


只要不調用fun3,這三行代碼是不會讓Python解譯器報錯的,因為我們已經知道,它等價於:
複製代碼 代碼如下:

def fun3(x):
pass
fun3 = entrance(fun3)



我們定義了一個帶參的函數fun3,然後把fun3指向了另一個函數inner(),當然不會有什麼錯。

但是,當我們使用fun3時,我們肯定會按照它定義時的樣子去使用它,給它傳入一個參數。
>>>fun3(1)
這裡就會出錯了,看看解譯器怎麼報錯的

Traceback (most recent call last):
File “decorator.py”, line 23, in www.jb51.net
fun3(1)
TypeError: inner() takes no arguments (1 given)

當然我們已經很容易知到為什麼會這樣報錯了,fun3已經不是指向它定義時那個函數了,它現在指向了”inner()”,而inner是沒有參數的,當然會出錯。
那怎麼解決呢?
修改一下inner()的定義,讓它可以就收任意個參數就可以了。

複製代碼 代碼如下:

def entrance(func):
def inner(*args, **kvargs):
print "inside function : ", func.__name__
func(*args, **kvargs)
return inner


現在,給inner傳任意個參數都不會出錯了,也就是entrance可以被用來裝飾任何一個函數了。

3. 寫個裝飾器logger

一個函數被調用時,在日誌裡記錄其名稱和被調用的實際參數
複製代碼 代碼如下:

def logger(func):
def inner(*args, **kvargs):
print func.__name__, 'called, arguments: ', args, kvargs
func(*args, **kvargs)
return inner

希望本文所述對大家的Python程式設計有所協助。

  • 相關文章

    聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.