您好,登錄后才能下訂單哦!
本篇文章為大家展示了如何在Python中定義裝飾器,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
Python 裝飾器
一、何為裝飾器
1、在函數中定義函數
在函數中定義另外的函數,就是說可以創建嵌套的函數,例子如下
def sayHi(name="hjj2"): print 'inside sayHi() func' def greet(): return 'inside greet() func' print(greet()) sayHi() #output # inside sayHi() func # inside greet() func
2、將函數作為參數傳給另外一個函數,裝飾器原型
def sayHi(): return 'hi hjj2' def doSthBeforeSayHi(func): print 'before sayHi func' print(func()) doSthBeforeSayHi(sayHi) #output # before sayHi func # hi hjj2
3、實現一個裝飾器
在第二步中,我們已經基本探究到裝飾器的原理了,python裝飾器做的事就是通過封裝一個函數并且用這樣或那樣的方式來修改它的行為。不帶@的初步示例如下:
def new_decorator(func): def wrapDecorator(): print 'before func' func() print 'after func' return wrapDecorator def func_require_decorator(): print 'a func need decorator' func_require_decorator() #ouput: a func need decorator func_require_decorator = new_decorator(func_require_decorator) func_require_decorator() #ouput: # before func # a func need decorator # after func
使用@來運行裝飾器
@new_decorator func_require_decorator() #ouput: # before func # a func need decorator # after func
這里我們可以看到,這兩個例子的運行結果是一樣的。所以我們能想象得到@new_decorator的作用就是
func_require_decorator = new_decorator(func_require_decorator)
我們繼續優化這個裝飾器,現在我們有一個問題就是,如果我們想要通過print(func_require_decorator.__name__)
就會報錯# Output: wrapTheFunction。這樣就需要借助python提供的functools.wraps
來解決了
@wraps接受一個函數來進行裝飾,并加入了復制函數名稱、注釋文檔、參數列表等等的功能。這可以讓我們在裝飾器里面訪問在裝飾之前的函數的屬性。
from functools import wraps def new_decorator(func): @wraps(func) def wrapDecorator(): print 'before func' func() print 'after func' return wrapDecorator def func_require_decorator(): print 'a func need decorator' @new_decorator func_require_decorator() print(func_require_decorator.__name__) #ouput: func_require_decorator
二、使用場景
1、授權,大體例子
from functools import wraps def requires_auth(f): @wraps(f) def decorated(*args, **kwargs): auth = request.authorization if not auth or not check_auth(auth.username, auth.password): authenticate() return f(*args, **kwargs) return decorated
2、日志:
from functools import wraps def logit(logfile='out.log'): def logging_decorator(func): @wraps(func) def wrapped_function(*args,**kwargs): log_string = func.__name__+"was called" print(log_string) with open(logfile,'a') as opened_file: opened_file.write(log_string+'\n') return func(*args,**kwargs) return wrapped_function return logging_decorator @logit() def func1(): pass func1()
3、其他如flask中的@app.route()
三、裝飾器類
1、將上面的日志裝飾器變為類的初步模型如下
from functools import wraps class logit(object): def __init__(self, logfile='out.log'): self.logfile = logfile def __call__(self, func): @wraps(func) def wrapped_function(*args, **kwargs): log_string = func.__name__ + "was called" print(log_string) # 打開logfile并寫入 with open(self.logfile, 'a') as open_file: # 將日志寫到指定文件 open_file.write(log_string + '\n') # 發送一個通知 self.notify() return func(*args, **kwargs) return wrapped_function def notify(self): pass @logit() def myfunc1(): pass class email_logit(logit): ''' 實現在函數調用時發送email ''' def __init__(self, email='admin@xxx.com', *args, **kwargs): self.email = email super(email_logit, self).__init__(*args, **kwargs) def notify(self): ''' 發送郵件通知 ''' pass
python的數據類型:1. 數字類型,包括int(整型)、long(長整型)和float(浮點型)。2.字符串,分別是str類型和unicode類型。3.布爾型,Python布爾類型也是用于邏輯運算,有兩個值:True(真)和False(假)。4.列表,列表是Python中使用最頻繁的數據類型,集合中可以放任何數據類型。5. 元組,元組用”()”標識,內部元素用逗號隔開。6. 字典,字典是一種鍵值對的集合。7. 集合,集合是一個無序的、不重復的數據組合。
上述內容就是如何在Python中定義裝飾器,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。