溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎么創(chuàng)建自己的Python裝飾器

發(fā)布時(shí)間:2022-09-05 09:27:54 來(lái)源:億速云 閱讀:117 作者:iii 欄目:開發(fā)技術(shù)

本文小編為大家詳細(xì)介紹“怎么創(chuàng)建自己的Python裝飾器”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“怎么創(chuàng)建自己的Python裝飾器”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來(lái)學(xué)習(xí)新知識(shí)吧。

1、@staticmethod

@staticmethod是python開發(fā)者經(jīng)常用來(lái)在一個(gè)類中聲明該函數(shù)是一個(gè)靜態(tài)函數(shù)時(shí)使用到的裝飾器,比如創(chuàng)建一個(gè)HelloWorld的python類,并且在其中使用該靜態(tài)裝飾器聲明其中的函數(shù)。

class HelloWorld():
    def __init__(self):
        super(HelloWorld, self).__init__()

    @staticmethod
    def print_hello_world():
        print('welcome to hello world!')

@staticmethod裝飾器一般是對(duì)于一些公共的函數(shù),或是工具函數(shù)之類的函數(shù)進(jìn)行聲明,聲明之后就不會(huì)將當(dāng)前python類中的初始化變量信息等傳入到該函數(shù)中,可以看到print_hello_world函數(shù)并沒(méi)有self作為參數(shù)變量。

接下來(lái)可以在初始化@staticmethod聲明的函數(shù)所在的類HelloWorld,并且調(diào)用print_hello_world函數(shù)。

hello_world = HelloWorld()
hello_world.print_hello_world()

會(huì)發(fā)現(xiàn)控制臺(tái)直接打印出了welcome to hello world!這行字符串。

實(shí)際上在python中的函數(shù)上面加入裝飾器只是為了在執(zhí)行當(dāng)前函數(shù)的邏輯之前去執(zhí)行一些我們需要執(zhí)行的業(yè)務(wù)功能,這樣的操作我們通過(guò)自定義自己的裝飾器也能夠?qū)崿F(xiàn)同樣的效果。

2、自定義裝飾器

其實(shí),自定義裝飾器的過(guò)程也比較簡(jiǎn)單,就是我們平常用到的函數(shù)或者python類的寫法就能夠?qū)崿F(xiàn)。

自己實(shí)現(xiàn)裝飾器主要有兩種方式,一種是通過(guò)class類的方式來(lái)實(shí)現(xiàn)的,另外一種則是通過(guò)python函數(shù)嵌套的方式來(lái)實(shí)現(xiàn)的,下面我們先來(lái)通過(guò)第一種的方式來(lái)實(shí)現(xiàn),也就是通過(guò)python類的方式來(lái)實(shí)現(xiàn)。

python類實(shí)現(xiàn)裝飾器

用python類來(lái)實(shí)現(xiàn)裝飾器時(shí),必須明白一個(gè)知識(shí)點(diǎn)。python類中實(shí)際上默認(rèn)有一個(gè)成員函數(shù)__call__,這個(gè)成員函數(shù)就是這個(gè)類被調(diào)用時(shí)的函數(shù)對(duì)象,若是需要自定義裝飾器實(shí)際上就是在python類的__call__函數(shù)中來(lái)實(shí)現(xiàn)裝飾器主要業(yè)務(wù)邏輯實(shí)現(xiàn)的。

class print_message(object):
    def __init__(self, function_):
        self.function_ = function_

    def __call__(self):
        # TODO:這里實(shí)際上是對(duì)傳入的函數(shù)返回值進(jìn)行的裝飾,可以理解成是一種回調(diào)。
        print('裝飾器,{}'.format(self.function_()))

注意:在下面這行代碼塊中一定要注意self.function_是一個(gè)函數(shù)對(duì)象,而self.function_()是一個(gè)函數(shù)返回值得效果。

print('裝飾器,{}'.format(self.function_()))

這樣,我們通過(guò)python類就已經(jīng)實(shí)現(xiàn)了python裝飾器的效果,使用一個(gè)函數(shù)來(lái)試驗(yàn)一下效果。

@print_message
def hello_world():
    return 'hello world!'

hello_world()

調(diào)用使用了@print_message裝飾器的函數(shù),它會(huì)返回我們預(yù)期的一個(gè)函數(shù)結(jié)果的打印。

# 裝飾器,hello world!

python函數(shù)嵌套實(shí)現(xiàn)裝飾器

上面的操作過(guò)程是通過(guò)重新定義了一個(gè)python類來(lái)實(shí)現(xiàn)裝飾器的效果的,這里再使用函數(shù)嵌套的方式來(lái)實(shí)現(xiàn)。

因?yàn)?,我們都知道在python中函數(shù)中再可以嵌套函數(shù)的,在函數(shù)中嵌套一個(gè)函數(shù)時(shí)上層的函數(shù)相對(duì)于子函數(shù)來(lái)說(shuō)就是它的一個(gè)父級(jí)對(duì)象。

@print_message2
@print_message
def hello_world3():
    return 'hello world!'

hello_world3()

# 裝飾器,hello world!
# 裝飾器2,None

使用函數(shù)嵌套的方式同樣實(shí)現(xiàn)了函數(shù)的裝飾器的效果,那么思考一下若是有兩個(gè)裝飾器可以在同一個(gè)函數(shù)中使用嗎?

多個(gè)裝飾器調(diào)用

話不多說(shuō),為了證明兩個(gè)裝飾器能不能放到一個(gè)函數(shù)上面使用,我們直接試一下效果如何。

@print_message2
@print_message
def hello_world3():
    return 'hello world!'

hello_world3()

# 裝飾器,hello world!
# 裝飾器2,None

從返回結(jié)果來(lái)看,首先是兩個(gè)裝飾器都是執(zhí)行了,從數(shù)據(jù)結(jié)果打印的順序來(lái)看自定義的裝飾器的執(zhí)行順序應(yīng)該是從距離函數(shù)最近的裝飾器開始執(zhí)行的,也就是從下往上的順序挨個(gè)執(zhí)行該函數(shù)上面的裝飾器的.

另外,裝飾器2的結(jié)果為None,這是為什么呢?

因?yàn)?第一個(gè)裝飾器執(zhí)行的時(shí)候,它的參數(shù)應(yīng)該是hello_world函數(shù)本身,但是當(dāng)?shù)诙€(gè)裝飾器執(zhí)行的時(shí)候第一個(gè)裝飾器并沒(méi)有返回結(jié)果知識(shí)做了打印,這個(gè)時(shí)候第二個(gè)裝飾器接收到的參數(shù)自然就是None了。

3、帶參數(shù)的裝飾器

說(shuō)實(shí)話帶參數(shù)的裝飾器在python中我見(jiàn)到的不多,不多在java中幾乎只要是裝飾器都是可以加參數(shù)執(zhí)行的。

還是來(lái)介紹一下,算是屬于擴(kuò)展知識(shí)吧,既然已經(jīng)看到了這里,不妨再多掌握個(gè)小技能吧,哈哈~

我們使用pytgon嵌套的函數(shù)功能來(lái)實(shí)現(xiàn)這個(gè)帶參數(shù)的裝飾器吧,個(gè)人覺(jué)得這種方便一些。

def header(message):
    def decorator(function_):
        def wrapper():
            return '帶參數(shù)的裝飾器,參數(shù):{0},{1}'.format(message, function_())
        return wrapper
    return decorator

@header('Python 集中營(yíng)')
def hello_world4():
    return 'hello world!'

print(hello_world4())

# 帶參數(shù)的裝飾器,參數(shù):Python 集中營(yíng),hello world!

OK,帶參數(shù)的裝飾器果然生效了,給@header加上參數(shù)@header(‘Python 集中營(yíng)’),上面裝飾器直接使用三層函數(shù)的嵌套來(lái)實(shí)現(xiàn)的。

第一層函數(shù)參數(shù)是我們需要自定義給裝飾器傳入的參數(shù),第二層則是傳入的已經(jīng)添加了裝飾器的函數(shù)本身,到了第三層才是真正的處理裝飾器自己的業(yè)務(wù)邏輯的。

讀到這里,這篇“怎么創(chuàng)建自己的Python裝飾器”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識(shí)點(diǎn)還需要大家自己動(dòng)手實(shí)踐使用過(guò)才能領(lǐng)會(huì),如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI