網站首頁 編程語言 正文
python裝飾器在平常的python編程中用到的還是很多的,在本篇文章中我們先來介紹一下python中最常使用的@staticmethod裝飾器的使用。
之后,我們會使用兩種不同的方式來創建自己的自定義python裝飾器以及如何在其他地方進行調用。
1、@staticmethod
@staticmethod是python開發者經常用來在一個類中聲明該函數是一個靜態函數時使用到的裝飾器,比如創建一個HelloWorld的python類,并且在其中使用該靜態裝飾器聲明其中的函數。
class HelloWorld():
def __init__(self):
super(HelloWorld, self).__init__()
@staticmethod
def print_hello_world():
print('welcome to hello world!')
@staticmethod裝飾器一般是對于一些公共的函數,或是工具函數之類的函數進行聲明,聲明之后就不會將當前python類中的初始化變量信息等傳入到該函數中,可以看到print_hello_world函數并沒有self作為參數變量。
接下來可以在初始化@staticmethod聲明的函數所在的類HelloWorld,并且調用print_hello_world函數。
hello_world = HelloWorld()
hello_world.print_hello_world()
會發現控制臺直接打印出了welcome to hello world!這行字符串。
實際上在python中的函數上面加入裝飾器只是為了在執行當前函數的邏輯之前去執行一些我們需要執行的業務功能,這樣的操作我們通過自定義自己的裝飾器也能夠實現同樣的效果。
2、自定義裝飾器
其實,自定義裝飾器的過程也比較簡單,就是我們平常用到的函數或者python類的寫法就能夠實現。
自己實現裝飾器主要有兩種方式,一種是通過class類的方式來實現的,另外一種則是通過python函數嵌套的方式來實現的,下面我們先來通過第一種的方式來實現,也就是通過python類的方式來實現。
python類實現裝飾器
用python類來實現裝飾器時,必須明白一個知識點。python類中實際上默認有一個成員函數__call__,這個成員函數就是這個類被調用時的函數對象,若是需要自定義裝飾器實際上就是在python類的__call__函數中來實現裝飾器主要業務邏輯實現的。
class print_message(object):
def __init__(self, function_):
self.function_ = function_
def __call__(self):
# TODO:這里實際上是對傳入的函數返回值進行的裝飾,可以理解成是一種回調。
print('裝飾器,{}'.format(self.function_()))
注意:在下面這行代碼塊中一定要注意self.function_是一個函數對象,而self.function_()是一個函數返回值得效果。
print('裝飾器,{}'.format(self.function_()))
這樣,我們通過python類就已經實現了python裝飾器的效果,使用一個函數來試驗一下效果。
@print_message
def hello_world():
return 'hello world!'
hello_world()
調用使用了@print_message裝飾器的函數,它會返回我們預期的一個函數結果的打印。
# 裝飾器,hello world!
python函數嵌套實現裝飾器
上面的操作過程是通過重新定義了一個python類來實現裝飾器的效果的,這里再使用函數嵌套的方式來實現。
因為,我們都知道在python中函數中再可以嵌套函數的,在函數中嵌套一個函數時上層的函數相對于子函數來說就是它的一個父級對象。
@print_message2
@print_message
def hello_world3():
return 'hello world!'
hello_world3()
# 裝飾器,hello world!
# 裝飾器2,None
使用函數嵌套的方式同樣實現了函數的裝飾器的效果,那么思考一下若是有兩個裝飾器可以在同一個函數中使用嗎?
多個裝飾器調用
話不多說,為了證明兩個裝飾器能不能放到一個函數上面使用,我們直接試一下效果如何。
@print_message2
@print_message
def hello_world3():
return 'hello world!'
hello_world3()
# 裝飾器,hello world!
# 裝飾器2,None
從返回結果來看,首先是兩個裝飾器都是執行了,從數據結果打印的順序來看自定義的裝飾器的執行順序應該是從距離函數最近的裝飾器開始執行的,也就是從下往上的順序挨個執行該函數上面的裝飾器的.
另外,裝飾器2的結果為None,這是為什么呢?
因為,第一個裝飾器執行的時候,它的參數應該是hello_world函數本身,但是當第二個裝飾器執行的時候第一個裝飾器并沒有返回結果知識做了打印,這個時候第二個裝飾器接收到的參數自然就是None了。
3、帶參數的裝飾器
說實話帶參數的裝飾器在python中我見到的不多,不多在java中幾乎只要是裝飾器都是可以加參數執行的。
還是來介紹一下,算是屬于擴展知識吧,既然已經看到了這里,不妨再多掌握個小技能吧,哈哈~
我們使用pytgon嵌套的函數功能來實現這個帶參數的裝飾器吧,個人覺得這種方便一些。
def header(message):
def decorator(function_):
def wrapper():
return '帶參數的裝飾器,參數:{0},{1}'.format(message, function_())
return wrapper
return decorator
@header('Python 集中營')
def hello_world4():
return 'hello world!'
print(hello_world4())
# 帶參數的裝飾器,參數:Python 集中營,hello world!
OK,帶參數的裝飾器果然生效了,給@header加上參數@header(‘Python 集中營’),上面裝飾器直接使用三層函數的嵌套來實現的。
第一層函數參數是我們需要自定義給裝飾器傳入的參數,第二層則是傳入的已經添加了裝飾器的函數本身,到了第三層才是真正的處理裝飾器自己的業務邏輯的。
原文鏈接:https://blog.csdn.net/chengxuyuan_110/article/details/126682396
相關推薦
- 2022-09-28 Dephi逆向工具Dede導出函數名MAP導入到IDA中的實現方法_python
- 2022-09-03 pandas?如何保存數據到excel,csv_python
- 2022-08-02 Golang中panic與recover的區別_Golang
- 2022-04-09 python的函數參數你了解嗎_python
- 2022-08-26 C++中Boost的智能指針scoped_ptr_C 語言
- 2023-01-12 python如何批量讀取.mat文件并保存成.npy_python
- 2022-07-06 Flutter?點擊兩次退出app的實現示例_Android
- 2022-10-11 RabbitMQ:生產者消息確認、消息持久化、消費者消息確認、消費失敗重試機制
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支