聊聊Python中的@符號(hào)是什么意思
Python中的@符號(hào)是裝飾器的意思。Python中裝飾器本質(zhì)上就是一個(gè)函數(shù),它可以讓其他函數(shù)在不需要做任何代碼變動(dòng)的前提下增加額外的功能,裝飾器的返回值也是一個(gè)函數(shù)對(duì)象(函數(shù)的指針)。
- 實(shí)質(zhì):是一個(gè)函數(shù)
- 參數(shù):是你要裝飾的函數(shù)名(并非函數(shù)調(diào)用)
- 返回:是裝飾完的函數(shù)名(也不是函數(shù)調(diào)用)
- 作用:為已經(jīng)存在的對(duì)象添加額外的功能。
- 特點(diǎn):不需要對(duì)對(duì)象做任何的代碼上的變動(dòng)。
Python裝飾器有很多經(jīng)典的應(yīng)用場(chǎng)景,比如:插入日志,性能測(cè)試,事務(wù)處理,權(quán)限校驗(yàn)等。裝飾器是解決這類(lèi)問(wèn)題的絕佳設(shè)計(jì)。
裝飾器最大的作用就是對(duì)于我們已經(jīng)寫(xiě)好的程序,我們可以抽離出一些雷同的代碼組件多個(gè)特定的裝飾器,這樣我們就可以針對(duì)不同的需求去使用特定的裝飾器,這時(shí),因?yàn)樵创a去除了大量泛化的內(nèi)容而使得源碼具有更加清晰的邏輯。
定義一個(gè)能打印日志的doctorator:
def log(func): def wrapper(*args, **kw): print('call %s():' % func.__name__) return func(*args, **kw) return wrapper @log def now(): print('2021-3-25') if __name__ == '__main__': now()
執(zhí)行結(jié)果:
函數(shù)對(duì)象有一個(gè)__name__屬性,可以拿到函數(shù)的名字。
調(diào)用now()函數(shù),不僅會(huì)運(yùn)行now()函數(shù)本身,還會(huì)在運(yùn)行now()函數(shù)前打印一行日志。
把@log放到now()函數(shù)的定義處,相當(dāng)于執(zhí)行了語(yǔ)句:
now = log(now)
wrapper()函數(shù)的參數(shù)定義是(*args,**kw),因此,wrapper()函數(shù)可以接收任意參數(shù)的調(diào)用。在wrapper()函數(shù)內(nèi),首先打印日志,再緊接著調(diào)用原始函數(shù)。
如果decorator本身需要傳入?yún)?shù),那就需要編寫(xiě)一個(gè)返回decorator的高階函數(shù):
def log(text): def decorator(func): def wrapper(*args, **kw): print('%s %s():' % (text, func.__name__)) return func(*args, **kw) return wrapper return decorator @log('execute') def now(): print('2015-3-25') if __name__ == '__main__': now()
執(zhí)行結(jié)果:
和兩層嵌套的decorator相比,3層嵌套的效果是這樣的:
now = log('execute')(now)
首先執(zhí)行l(wèi)og(‘execute'),返回的是decorator函數(shù),再調(diào)用返回的函數(shù),參數(shù)就是now函數(shù),返回值最終是wrapper函數(shù)。
函數(shù)也是對(duì)象,它有__name_等屬性,但你去看經(jīng)過(guò)decorator裝飾之后的函數(shù),他們的__name__已經(jīng)從原來(lái)的'now'變成了'wrapper':
print(now.__name__) #輸出:wrapper
因?yàn)榉祷氐哪莻€(gè)wrapper()函數(shù)名字就是'wrapper',所以,需要把原始函數(shù)的__name__等屬性賦值到wrapper()函數(shù)中,否則,有些依賴(lài)函數(shù)簽名的代碼執(zhí)行就會(huì)出錯(cuò)。
Python內(nèi)置的functiontools.wraps就是干這個(gè)事的,所以,一個(gè)完整的decorator的寫(xiě)法如下:
import functools def log(func): @functools.wraps(func) def wrapper(*args, **kw): print('call %s():' % func.__name__) return func(*args, **kw) return wrapper
或者
import functools def log(text): def decorator(func): @functools.wraps(func) def wrapper(*args, **kw): print('%s %s():' % (text, func.__name__)) return func(*args, **kw) return wrapper return decorator
總結(jié)
到此這篇關(guān)于Python中@符號(hào)是什么意思的文章就介紹到這了,更多相關(guān)Python中@符號(hào)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python數(shù)據(jù)可視化實(shí)現(xiàn)正態(tài)分布(高斯分布)
這篇文章主要介紹了Python數(shù)據(jù)可視化實(shí)現(xiàn)正態(tài)分布(高斯分布),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08python+requests實(shí)現(xiàn)接口測(cè)試的完整步驟
這篇文章主要給大家介紹了關(guān)于python+requests實(shí)現(xiàn)接口測(cè)試的完整步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10django rest framework serializer返回時(shí)間自動(dòng)格式化方法
這篇文章主要介紹了django rest framework serializer返回時(shí)間自動(dòng)格式化方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Python神器之使用watchdog監(jiān)控文件變化
這篇文章主要為大家詳細(xì)介紹了Python中的神器watchdog以及如何使用watchdog監(jiān)控文件變化,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解下2023-12-12python網(wǎng)絡(luò)爬蟲(chóng)基本語(yǔ)法詳解
掌握Python網(wǎng)絡(luò)爬蟲(chóng)基本語(yǔ)法,就是打開(kāi)數(shù)據(jù)世界的鑰匙,在這份指南中,我們將帶你深入淺出,從零開(kāi)始一步步變成抓取信息的高手,準(zhǔn)備好探索無(wú)限可能的網(wǎng)絡(luò)數(shù)據(jù)世界了嗎?讓我們一起開(kāi)啟這段精彩旅程吧!2024-03-03python開(kāi)發(fā)游戲的前期準(zhǔn)備
在本篇文章中我們給大家分享了關(guān)于python開(kāi)發(fā)游戲的前期準(zhǔn)備以及用到的工具等內(nèi)容,需要的朋友們跟著參考下。2019-05-05python tkinter GUI繪制,以及點(diǎn)擊更新顯示圖片代碼
這篇文章主要介紹了python tkinter GUI繪制,以及點(diǎn)擊更新顯示圖片代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Python中集合的內(nèi)建函數(shù)和內(nèi)建方法學(xué)習(xí)教程
這篇文章主要介紹了Python中集合的內(nèi)建函數(shù)和內(nèi)建方法學(xué)習(xí)教程,包括工廠(chǎng)函數(shù)和僅用于可變集合的方法等知識(shí)點(diǎn),需要的朋友可以參考下2015-08-08