主頁 > 知識(shí)庫 > python基礎(chǔ)之裝飾器詳解

python基礎(chǔ)之裝飾器詳解

熱門標(biāo)簽:地圖標(biāo)注微信發(fā)送位置不顯示 地圖制圖標(biāo)注位置改變是移位嗎 南京銷售外呼系統(tǒng)軟件 房產(chǎn)電銷外呼系統(tǒng) 315電話機(jī)器人廣告 蓋州市地圖標(biāo)注 上海機(jī)器人外呼系統(tǒng)哪家好 浙江電銷卡外呼系統(tǒng)好用嗎 地圖標(biāo)注的意義點(diǎn)

一、前言

裝飾器:本質(zhì)就是函數(shù),功能是為其他函數(shù)添加附加功能

原則:

  •     1、不修改被修飾函數(shù)的源代碼
  •     2、不修改被修飾函數(shù)的調(diào)用方式

裝飾器 = 高階函數(shù) + 函數(shù)嵌套 + 閉包

二、高階函數(shù)

高階函數(shù)定義:

  •     1、函數(shù)接收的參數(shù)是一個(gè)函數(shù)
  •     2、函數(shù)的返回值是一個(gè)函數(shù)名
  •     3、滿足上述條件任意一個(gè),都可以稱為高階函數(shù)

test 函數(shù)是高階函數(shù),接受了一個(gè)foo 作為參數(shù)

import time
def foo():
    time.sleep(3)
    print("sleep 3s")
 
def test(func):
    start_time = time.time()
    func()
    stop_time = time.time()
    print("函數(shù)的運(yùn)行時(shí)間是: %s" % (stop_time - start_time))
 
test(foo)

timer 是一個(gè)高階函數(shù),這個(gè)函數(shù)返回值是一個(gè)函數(shù)

import time
def foo():
    time.sleep(3)
    print("sleep 3s")
 
def timer(func):
    start_time = time.time()
    func()
    stop_time = time.time()
    print("執(zhí)行時(shí)間{}".format(stop_time - start_time))
    return func
foo = timer(foo)
foo()
# 結(jié)果: 多運(yùn)行了一次

三、函數(shù)嵌套

在函數(shù)里面定義函數(shù),變量的作用域和生存周期不變。

def father(name):
    print("father name: %s" % name)
    def son():
        print("son name: %s" % name)
    son()
father("xu1")
 
# 結(jié)果:
#     father name: xu1
#     son name: xu1

四、裝飾器

實(shí)現(xiàn)一個(gè)計(jì)算函數(shù)執(zhí)行時(shí)間的函數(shù)作為裝飾器,用來計(jì)算被裝飾函數(shù)的執(zhí)行時(shí)間并打印

import time
 
def timer(func):  # 實(shí)現(xiàn)一個(gè)計(jì)算函數(shù)執(zhí)行時(shí)間的函數(shù)作為裝飾器,用來計(jì)算被裝飾函數(shù)的執(zhí)行時(shí)間并打出
    def wrapper():
        start_time = time.time()
        func()
        stop_time = time.time()
        print("運(yùn)行時(shí)間: %s" % (stop_time - start_time))
    return wrapper
 
# def test():  # 不使用裝飾器的同等實(shí)現(xiàn)
#     time.sleep(3)
#     print("test sleep 3s")
#
# test = timer(test)  # 返回的是 wrapper 的地址
# test()  # 執(zhí)行的是 wrapper
 
 
@timer
def test():  # 裝飾器的實(shí)現(xiàn)
    time.sleep(3)
    print("test sleep 3s")
 
test()  # 執(zhí)行的是 wrapper
# 結(jié)果:
#     test sleep 3s
#     運(yùn)行時(shí)間: 3.000915050506592

4.1 被裝飾方法帶返回值

import time
 
 
def timer(func):
    def wrapper():
        start_time = time.time()
        res = func()  # 執(zhí)行被裝飾方法
        stop_time = time.time()
        print("運(yùn)行時(shí)間: %s" % (stop_time - start_time))
        return res  # 接受正在調(diào)用的方法的返回值,并返回
    return wrapper
 
 
@timer
def test():
    time.sleep(3)
    print("test sleep 3s")
    return "test return ok"
 
 
print(test())  # 執(zhí)行的是 wrapper
# 結(jié)果:
#     test sleep 3s
#     運(yùn)行時(shí)間: 3.0002923011779785
#     test return ok

4.2 被裝飾方法帶參數(shù)

import time
 
 
def timer(func):
    """
        *args:將被修飾方法傳入的非關(guān)鍵字參數(shù)打包為元組 args
        **kwargs: 將被修飾方法傳入的關(guān)鍵字參數(shù)打包為字典 kwargs
    """
    def wrapper(*args, **kwargs):
        start_time = time.time()
        res = func(*args, **kwargs)  # *args 拆解元組,按順序傳給被修飾函數(shù); **kwargs:拆解字典
        stop_time = time.time()
        print("運(yùn)行時(shí)間: %s" % (stop_time - start_time))
        return res
    return wrapper
 
 
@timer  # 給test 方法添加計(jì)算執(zhí)行時(shí)間的裝飾器
def test(name, age):
    time.sleep(3)
    print("name = {}, age = {}".format(name, age))
    return "test return ok"
 
 
# 調(diào)用被裝飾器裝飾的方法
print(test("xu", 100))  # 執(zhí)行的是 wrapper
# 結(jié)果:
#     name = xu, age = 100
#     運(yùn)行時(shí)間: 3.000420331954956
#     test return ok

4.3 驗(yàn)證功能裝飾器

假如 index() 、home()、shopping_car() 三個(gè)方法都需要登錄后才能訪問(無法訪問時(shí)里面不輸入對(duì)應(yīng)內(nèi)容),正常情況下只需登錄一次,后面訪問其他方法就無需再次登錄。

可以通過@auth_fun裝飾器進(jìn)行驗(yàn)證用戶是否登錄,如果沒有就讓用戶輸入賬號(hào)密碼,用戶賬號(hào)密碼正確的記錄當(dāng)前登錄的用戶,其他方法無需再次登錄。

# 用戶列表
user_list = [
    {'name': 'xu1', 'passwd': '123'},
    {'name': 'xu2', 'passwd': '123'},
    {'name': 'xu3', 'passwd': '123'},
    {'name': 'xu4', 'passwd': '123'},
]
# 當(dāng)前登錄的用戶
current_dic = {"username": None, "login": False}
 
 
# 驗(yàn)證用戶是否登錄的裝飾器
#   如果用戶沒有登錄,讓用戶輸入賬號(hào)密碼,校驗(yàn)通過記錄用戶狀態(tài)
def auth_fun(func):
    def wrapper(*args, **kwargs):
        if current_dic["username"] and current_dic['login']:
            res = func(*args, **kwargs)
            return res
        username = input("請(qǐng)輸入用戶名:")
        pw = input("請(qǐng)輸入密碼:")
        for u in user_list:
            if u["name"] == username and u["passwd"] == pw:
                current_dic["username"] = username
                current_dic["login"] = True
                res = func(*args, **kwargs)
                return res
        else:
            print("用戶沒有注冊(cè)!")
    return wrapper
 
 
@auth_fun
def index():
    print("this is index")
 
 
@auth_fun
def home():
    print("this is home page")
 
 
@auth_fun
def shopping_car():
    print("this is shopping car")
 
 
index()  # 輸入用戶密碼
home()  # index 已經(jīng)登錄,無需在輸入
shopping_car()  # index 已經(jīng)登錄,無需在輸入
# 結(jié)果:
#     請(qǐng)輸入用戶名:xu1
#     請(qǐng)輸入密碼:123
#     this is index
#     this is home page
#     this is shopping car

4.4 驗(yàn)證功能裝飾器——帶參數(shù)

 裝飾器帶參數(shù),最簡單的操作就是可以對(duì)被裝飾的函數(shù)進(jìn)行區(qū)別處理。

# 用戶列表
user_list = [
    {'name': 'xu1', 'passwd': '123'},
    {'name': 'xu2', 'passwd': '123'},
    {'name': 'xu3', 'passwd': '123'},
    {'name': 'xu4', 'passwd': '123'},
]
# 當(dāng)前登錄的用戶
current_dic = {"username": None, "login": False}
 
"""
    注意:帶參數(shù)的裝飾器會(huì)比沒有帶參數(shù)的裝飾器多嵌套一層函數(shù)(多了auth)
        調(diào)用方式是 @auth(auth_type="type1"), 返回 auth_fun,
        也就是說 @auth(auth_type="type1")相當(dāng)于 @auth_fun
        但是 auth_fun 函數(shù)所在的嵌套作用域多了一個(gè) auth_type 的變量
"""
def auth(auth_type="type1"):
    def auth_fun(func):
        def wrapper(*args, **kwargs):
            if auth_type == "type1":
                if current_dic["username"] and current_dic['login']:
                    res = func(*args, **kwargs)
                    return res
                username = input("請(qǐng)輸入用戶名:")
                pw = input("請(qǐng)輸入密碼:")
                for u in user_list:
                    if u["name"] == username and u["passwd"] == pw:
                        current_dic["username"] = username
                        current_dic["login"] = True
                        res = func(*args, **kwargs)
                        return res
                else:
                    print("用戶沒有注冊(cè)!")
            elif auth_type == "type2":
                print("不用授權(quán)直接登錄: type = {}".format(auth_type))
                res = func(*args, **kwargs)
                return res
            else:
                print("其他type沒有實(shí)現(xiàn)")
        return wrapper
    return auth_fun
 
 
"""
    auth_fun = @auth(auth_type="type1") 
    auth_fun 所在的嵌套與將有一個(gè) auth_type 變量
    然后通過 @auth()方法返回的對(duì)象注解 index,相當(dāng)于 @auth_fun 注解index 方法,最后得到 wrapper 對(duì)象
"""
@auth(auth_type="type1")
def index():
    print("this is index")
 
 
@auth(auth_type="type2")
def home():
    print("this is home page")
 
 
@auth(auth_type="type3")
def shopping_car():
    print("this is shopping car")
 
 
home()  # 注意:auth_type="type2",這個(gè)方法無需登錄可以直接執(zhí)行
index()  # 注意:auth_type="type1",需要登錄
shopping_car()  # 注意:auth_type="type3",沒有做處理
# 結(jié)果:
#     不用授權(quán)直接登錄: type = type2
#     this is home page
#     請(qǐng)輸入用戶名:xu1
#     請(qǐng)輸入密碼:123
#     this is index
#     其他type沒有實(shí)現(xiàn)

到此這篇關(guān)于python基礎(chǔ)之裝飾器詳解的文章就介紹到這了,更多相關(guān)python裝飾器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 詳解Python裝飾器之@property
  • python 裝飾器的使用與要點(diǎn)
  • python高級(jí)語法之閉包和裝飾器詳解
  • Python pytest裝飾器總結(jié)(實(shí)例詳解)
  • Python裝飾器的應(yīng)用場(chǎng)景及實(shí)例用法
  • Python 的lru_cache裝飾器使用簡介
  • Python 中@lazyprop 裝飾器的用法

標(biāo)簽:陽泉 金華 臨汾 日照 雙鴨山 克拉瑪依 貴州 赤峰

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《python基礎(chǔ)之裝飾器詳解》,本文關(guān)鍵詞  python,基礎(chǔ),之,裝飾,器,詳解,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《python基礎(chǔ)之裝飾器詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于python基礎(chǔ)之裝飾器詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章