中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

Python裝飾器實例應用分析

發布時間:2022-03-29 16:42:42 來源:億速云 閱讀:108 作者:iii 欄目:移動開發

本文小編為大家詳細介紹“Python裝飾器實例應用分析”,內容詳細,步驟清晰,細節處理妥當,希望這篇“Python裝飾器實例應用分析”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。

 一、函數回顧

1、在python中函數是一等公民,函數也是對象。我們可以把函數賦予變量。

def make_cofe(type):
    print("獲得一杯 : {}".format(type))

get_cofe = make_cofe
get_cofe("咖啡")

####輸出#####
獲得一杯 : 咖啡

這個例子中,我們把函數make_cofe 賦予了變量 get_cofe,這樣之后你調用 get_cofe,就相當于是調用函數 make_cofe()。

2、把函數當作參數,傳入另一個函數中。

def make_cofe(type):
    print("獲得一杯 : {}".format(type))

def shop(func,type):
    func(type)

shop(make_cofe,"咖啡")

####輸出####
獲得一杯 : 咖啡

這個例子,我們把make_cofe以參數的形式傳入shop中,然后調用它。

3、函數是可以嵌套的。

def shop(type):
    def make_cofe(type):
        print("獲得一杯 : {}".format(type))
    make_cofe(type)

shop("咖啡")

#####輸出####
獲得一杯 : 咖啡

這段代碼中,我們在函數shop內部定義了函數make_cofe 4、函數的返回值也可以是函數對象(閉包)。

def shop():
    def make_cofe(type):
        print("獲得一杯 : {}".format(type))
    return make_cofe

get_cofe=shop()
get_cofe("咖啡")

####輸出#####
獲得一杯 : 咖啡

這里,函數 shop() 的返回值是函數對象 make_cofe 本身,之后,我們將其賦予變量 get_cofe,再調用 get_cofe("咖啡")。

二、裝飾器

下面我們正式開始裝飾器的學習。 我們先想一個問題。如果我們去咖啡店要一杯咖啡,我們應該如何實現。你也許會這么寫。

def cofe():
    print("咖啡", end="")

cofe()

####輸出####
咖啡

那我們現在想來一杯加糖咖啡,我們該如何寫呢?你也許會這么想,那還不簡單,直接在cofe()函數里改不就好了。

def cofe():
    print("加糖咖啡", end="")

cofe()

####輸出####
加糖咖啡

那么問題來了,如果我們現在不想喝加糖咖啡了,該怎么辦呢,總不能在cofe()函數里去掉吧。那如果有人想喝加糖咖啡、有人不想喝加糖咖啡如何是好,總不能寫兩個cofe()函數吧。 那我們帶著問題看一下下面這段代碼。

def add_sugar(func):
    def add():
        print("加糖",end="")
        func()
    return add

def cofe():
    print("咖啡",end="")

cofe = add_sugar(cofe)

print("獲得一杯",end="")
cofe()

####輸出#####
獲得一杯加糖咖啡

變量 cofe 指向了內部函數 add(),而內部函數 add() 中又會調用原函數 cofe(),因此,最后調用 cofe() 時,就會先打印‘加糖",然后輸出‘咖啡"。這里的函數 add_sugar() 就是一個裝飾器,它把真正需要執行的函數cofe()包裹在其中,并且改變了它的行為,但是原函數 cofe() 不變。 下面我們來看一下更優雅的寫法。

def add_sugar(func):
    def add():
        print("加糖",end="")
        func()
    return add

@add_sugar
def cofe():
    print("咖啡",end="")

print("獲得一杯",end="")
cofe()

#####輸出#####
獲得一杯加糖咖啡

這里的@叫做語法糖, @add_sugar就相當于前面的cofe = add_sugar(cofe)語句,只不過更加簡潔。因此程序中建議用這種寫法。 好了,讓我們來回顧下我們的問題,如果有人想喝加糖咖啡、有人不想喝加糖咖啡如何是好。學了裝飾器那不就很簡單了,如果要喝加糖咖啡,我們把加糖的裝飾器@add_sugar給加上不就好了,如果喝不加糖的,那就不加裝飾器,這樣我們就把這個問題給完美解決掉了。在不改變函數內部的前提了,給函數又添加了新的功能。 到目前為止,我們已經把最簡單的裝飾器學完了。下面我們在考慮一個問題,如果原函數 cofe() 中,有參數需要傳遞給裝飾器怎么辦?一個簡單的辦法,是可以在對應的裝飾器函數 add() 上,加上相應的參數。

def add_sugar(func):
    def add(type):
        print("加糖",end="")
        func(type)
    return add

@add_sugar
def cofe(type):
    print("{}咖啡".format(type),end="")


cofe("美式")
print()
cofe("拿鐵")

####輸出#####
加糖美式咖啡
加糖拿鐵咖啡

不過,新的問題來了。如果我另外還有一個函數(奶茶函數),也需要使用 add_sugar() 裝飾器,但是這個新的函數有兩個參數,又該怎么辦呢? 通常情況下,我們會把*args和 **kwargs,作為裝飾器內部函數 add() 的參數。*args和**kwargs,表示接受任意數量和類型的參數,因此加糖裝飾器就可以寫成下面的形式:

def add_sugar(func):
    def add(*args, **kwargs):
        print("加糖",end="")
        func(*args, **kwargs)
    return add
     
@add_sugar
def cofe(type):
    print("{}咖啡".format(type),end="")

@add_sugar
def milk_tea(type,num):
    print("{}杯{}奶茶".format(num,type), end="")


cofe("美式")
print()
milk_tea("xx牌子","4")

####輸出####
加糖美式咖啡
加糖4杯xx牌子奶茶

這樣我們的咖啡和奶茶都可以加糖了。 前面我們講的是函數的裝飾器,下面我們來講一下類作為裝飾器。類裝飾器主要依賴于函數__call__(),每當你調用一個類的實例時,函數__call__()就會被執行一次。

class Add_sugar:
    def __init__(self, func):
        self.func = func
        self.add_suger = "加糖"

    def __call__(self, *args, **kwargs):
        print(self.add_suger,end="")
        return self.func(*args, **kwargs)

@Add_sugar
def cofe():
    print("咖啡")

cofe()

####輸出#####
加糖咖啡

最后如果我們的咖啡既要加糖又要加冰,那我們該如何做呢?我們定義一個加冰的裝飾器就好了呀。

def add_sugar(func):
    def add():
        print("加糖",end="")
        func()
    return add

def add_ice(func):
    def add():
        print("加冰",end="")
        func()
    return add


@add_sugar
@add_ice
def cofe():
    print("咖啡",end="")

cofe()

####輸出####
加糖加冰咖啡

讀到這里,這篇“Python裝飾器實例應用分析”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

静安区| 论坛| 牙克石市| 白玉县| 巴林右旗| 龙海市| 澄迈县| 屯门区| 呼图壁县| 韩城市| 内黄县| 东阿县| 安国市| 昌乐县| 信丰县| 丹寨县| 富锦市| 清远市| 贵德县| 伊春市| 武夷山市| 西吉县| 久治县| 革吉县| 孝义市| 莱芜市| 顺昌县| 仙居县| 盖州市| 交城县| 夏河县| 炉霍县| 孝感市| 兰溪市| 伊通| 柳州市| 长泰县| 巩留县| 烟台市| 大厂| 始兴县|