溫馨提示×

溫馨提示×

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

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

python裝飾器如何使用

發(fā)布時(shí)間:2020-09-24 11:25:37 來源:億速云 閱讀:105 作者:Leah 欄目:編程語言

這篇文章將為大家詳細(xì)講解有關(guān)python裝飾器如何使用,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。

裝飾者模式是常用的軟件設(shè)計(jì)模式之一。通過此設(shè)計(jì)模式,我們能夠在不修改任何底層代碼情況下,給已有對象賦予新的職責(zé)。python中可以用裝飾器簡單地實(shí)現(xiàn)裝飾者模式。

1.1 將函數(shù)作為參數(shù)傳遞

在C/C++中,函數(shù)指針可以將函數(shù)作為參數(shù)傳遞給另一函數(shù)。而在python中,函數(shù)也是對象的一種,函數(shù)可以被引用,也可直接作為參數(shù)傳入函數(shù),以及作為容器對象的元素。python中可以采用如下方法實(shí)現(xiàn)裝飾者模式:

#!/usr/bin/env python3.6
# -*- coding: utf-8 -*-

def add(x, y):
    result = x+y
    return result

def log(func):
    def wrapper(*args, **kwargs):
        result = func(*args)
        print(func.__name__,'has been called\n')
        return result
    return wrapper

if __name__ == '__main__':
    print(log(add)(1,2))

上述代碼中,log函數(shù)以需要被裝飾的函數(shù)作為參數(shù),并返回函數(shù)對象。被返回的函數(shù)的參數(shù)為可變參數(shù)*args與**kwargs(*args參數(shù)會被封裝成tuple,**kwargs參數(shù)則會被封裝成字典對象),以適應(yīng)不同函數(shù)的不同參數(shù),保證通用性。

1.2 裝飾器

上面的實(shí)現(xiàn)方法有些繁雜,所有調(diào)用被裝飾的函數(shù)之處的代碼,都要進(jìn)行相應(yīng)修改,自然不符合python簡潔易讀的特性。因此python中給出相應(yīng)語法糖來增加可讀性和易用性,那便是“裝飾器”。

from functools import wraps

def log(func):
    #@wraps(func)
    def wrapper(*args, **kwargs):
        result = func(*args)
        print(func.__name__,'has been called')
        return result
    return wrapper

#等價(jià)于add = log(add)
@log
def add(x, y):
    result = x+y
    return result

if __name__ == '__main__':
    print(add(1,2))
    print(add.__name__)

運(yùn)行情況如下:

>>print(add(1,2))
add has been called
3
>>print(add.__name__)
wrapper

但上述方法亦有缺陷,原函數(shù)add的元數(shù)據(jù)(比如名字、文檔字符串、注解和參數(shù)簽名)會丟失。為避免缺陷,任何時(shí)候你定義裝飾器的時(shí)候,都應(yīng)該使用functools庫中的@wraps裝飾器來注解底層包裝函數(shù)(代碼中注釋部分)。@wraps有一個(gè)重要特征是它能讓你通過屬性 __wrapped__ 直接訪問被包裝函數(shù)。

改進(jìn)后運(yùn)行情況:

>>print(add(1,2))
add has been called
3
>>print(add.__name__)
add

1.3 解除裝飾器

當(dāng)裝飾器已經(jīng)作用于某函數(shù),而你想撤銷它,那么可以訪問 __wrapped__屬性來訪問原始函數(shù)

orig_add = add.__wrapped__
orig_add(1,2)

但若使用了多個(gè)裝飾器, __wrapped__屬性會變得不可控,應(yīng)盡量避免使用。
若有如下代碼:

#!/usr/bin/env python3.6
# -*- coding: utf-8 -*-

import functools
import time

def metric(func):
    @functools.wraps(func)
    def wrapper(*args,**kv):
        print('Decorator1')
        f = func(*args,**kv)
        return f
    return wrapper

def logging(func):
    @functools.wraps(func)
    def wrapper(*args,**kv):
        print('Decorator2')
        f = func(*args,**kv)
        return f
    return wrapper

@metric
@logging
def normalize(name):
    sName = name[0:1].upper() + name[1:].lower()
    print(sName)

if __name__ == '__main__':
    normalize('heLlO')
    normalize.__wrapper__('')

運(yùn)行情況如下:

>>normalize('helLo')
Decorator1
Decorator2
Hello
>>normalize.__wrapped__('world')
Decorator2
World

1.4 定義帶參數(shù)的裝飾器

from functools import wraps

def log(text):
    def decorator(func):
        @wraps(func) 
        def wrappering(*args,**kv):
            print('%s %s():'%(text,func.__name__))
            return func(*args,**kv)
        return wrappering
    return decorator

@log('run')
def normalize(name):
    sName = name[0:1].upper() + name[1:].lower()
    print(sName)

裝飾器函數(shù)可以帶參數(shù),最外層的函數(shù)會將參數(shù)傳給內(nèi)層的裝飾器函數(shù),即wrappering函數(shù)是可以使用log的傳入?yún)?shù)的。

裝飾器處理過程與下面是等價(jià)的:

normalize = log('run')(normalize)

關(guān)于python裝飾器如何使用就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

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

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

AI