亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

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

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

Python無參裝飾器的實現方法

發布時間:2021-08-16 15:40:11 來源:億速云 閱讀:117 作者:chen 欄目:開發技術

本篇內容主要講解“Python無參裝飾器的實現方法”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“Python無參裝飾器的實現方法”吧!

目錄
  • 一、什么是裝飾器

  • 二、何時用裝飾器

  • 三、如何寫一個裝飾器

    • 方案一:

    • 方案二:

    • 方案三:

    • 優化一(參數優化,實現任意參數): 

    • 優化二(實現裝飾其他對象):

    • 優化三(得到相同返回值):

  •  總結: 

    一、什么是裝飾器

    定義一個函數,該函數可為其他函數添加額外的功能。

    二、何時用裝飾器

    需要在不修改被裝飾對象源代碼及其調用方式時,為被裝飾對象添加額外的功能。

    三、如何寫一個裝飾器

    現在我們有如下一個函數help(),time.sleep()來模擬函數執行時間,print打印傳入參數值,方便我們來進行分析。如果現在我們需要為help函數添加一個統計其運行時間的功能,我們可以怎么做?

    import time
     
    def help(x, y):
        time.sleep(1)
        print(f'x={x} y={y}')
     
    help(1, 2)

    方案一:

    在help函數開頭結束分別調用time.time(),兩者相減得運行時間。

    import time
     
    def help(x, y):
        start = time.time()
        time.sleep(1)
        print(f'x={x} y={y}')
        stop = time.time()
        print(stop - start)
     
    help(1, 2)

    運行結果:

    Python無參裝飾器的實現方法

    方案一中我們在help中加了相關代碼,雖然沒有改變它的調用方式,但改變了它的源代碼。我們繼續想想如何兩者都不改變的情況下,完成我們的目標。

    對,函數內不能動,我們可以動函數外呀,在help前后加上相關代碼,似乎就可以達到我們的目標了,這就是方案二,我們來試試。

    方案二:

    import time
     
    def help(x, y):
        time.sleep(1)
        print(f'x={x} y={y}')
     
    start = time.time()
    help(1, 2)
    stop = time.time()
    print(stop - start)

    運行結果:

    Python無參裝飾器的實現方法

    顯而易見,似乎沒有問題,但是如果我們需要多次調用help函數的話,在它前后都得加上相同的代碼,這樣代碼就會顯得十分冗余了。既然help函數前后代碼不會變的話,我們可以將它們封裝成另一個函數呀,說干就干。

    方案三:

    import time
     
    def help(x, y):
        time.sleep(1)
        print(f'x={x} y={y}')
     
    def wrapper():
        start = time.time()
        help(1, 2)
        stop = time.time()
        print(stop - start)
     
    wrapper()

    運行一下:

     Python無參裝飾器的實現方法

    這樣我們就解決了多次調用的問題,但美中不足的是,help函數的調用方式改變了,而且help的參數固定,也只能修飾help函數,我們來一步步試著優化。

    優化一(參數優化,實現任意參數): 

    對參數優化,我們可以將help的實參通過wrapper的傳入,而為了實現任意參數,我們首先想的便是*args,**kwargs來作為函數的參數,于是將方案三進行改進如下(為方便分析,為help多增加了一個參數):

    import time
     
     
    def help(x, y, z):
        time.sleep(1)
        print(f'x={x} y={y} z={z}')
     
     
    def wrapper(*args, **kwargs):
        start = time.time()
        help(*args, **kwargs)
        stop = time.time()
        print(stop - start)
     
     
    wrapper(1, 2, 3)

    運行一下:

    Python無參裝飾器的實現方法

    這樣我們便將help的參數變得更加靈活了,接著我們來優化。

    優化二(實現裝飾其他對象):

    需要裝飾其他對象,意味著我們在help位置的應該是一個可變參數,也就是用戶輸入的參數,即wapper函數內應變為:

    def wrapper(*args, **kwargs):
        start = time.time()
        func(*args, **kwargs)
        stop = time.time()
        print(stop - start)

    但是我們期望wrapper能和內部調用的func函數的參數一致,即wrapper的參數我們應該不去改變,那我們func的值從何處傳來呢?

    沒錯,我們可以運用閉包函數來傳參,修改一下下:

    def outter(func):
        def wrapper(*args, **kwargs):
            start = time.time()
            func(*args, **kwargs)
            stop = time.time()
            print(stop - start)
     
        return wrapper

    這樣我們為其他函數修飾時,只需要將其函數名作為outter函數的參數傳入即可:

    import time
     
    def help(x, y, z):
        time.sleep(1)
        print(f'這是help的{x}{y}{z}')
     
    def others(x, y, z):
        time.sleep(1)
        print(f'這是others的{x}{y}{z}')
     
    def outter(func):
        def wrapper(*args, **kwargs):
            start = time.time()
            func(*args, **kwargs)
            stop = time.time()
            print(stop - start)
     
        return wrapper
     
    help = outter(help)
    others = outter(others)
     
    help(1, 2, 3)
    others(4, 5, 6)

    運行一下:

    Python無參裝飾器的實現方法

    結果符合預期,而且在使用時由于outter內的func是在局部名稱空間,outter外的func是在全局名稱空間,調用時二者并不沖突,并且使用時可讀性較高,我們好像達成開始的目標,似乎能以假亂真了。但我們繼續思考一下,我們演示用到的函數十分簡單,甚至沒有返回值,如果加上返回值后,我們再對其修飾后,能得到原函數的返回值嗎?

    優化三(得到相同返回值):

    回到我們的wrapper中去,既然需要我們func函數的返回值,我們直接將其賦值給res,再return出res的值:

    import time
     
    def help(x, y, z):
        time.sleep(1)
        print(f'這是help的{x}{y}{z}')
        return 'help'
     
    def others(x, y, z):
        time.sleep(1)
        print(f'這是others的{x}{y}{z}')
        return 'others'
     
    def outter(func):
        def wrapper(*args, **kwargs):
            start = time.time()
            res=func(*args, **kwargs)
            stop = time.time()
            print(stop - start)
            return res
        return wrapper
     
    help = outter(help)
    others = outter(others)
     
    res1=help(1, 2, 3)
    res2=others(4, 5, 6)
    print(res1,res2)

    沒毛病,跑一下:

    Python無參裝飾器的實現方法

     總結: 

    到這我們完成了一個簡單的無參裝飾器,裝飾后的func既沒有改變源代碼,也沒有改變調用方式。

    但是代碼稍顯冗余,python語法便規定:在被裝飾對象正上方單獨一行寫@裝飾器名字,等價于func=outter(func),簡化代碼。從中我們總結出無參裝飾器的一個模板:

    def outter(func):
         def wrapper(*args,**kwargs):
             # 1、調用原函數
             # 2、增加的新功能
             res=func(*args,**kwargs)
             return res
         return wrapper
     
    #使用時
    @outter
    def func:
        pass

    到此,相信大家對“Python無參裝飾器的實現方法”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!

    向AI問一下細節

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

    AI

    闽侯县| 普兰县| 忻城县| 公安县| 大竹县| 电白县| 汤原县| 东乡族自治县| 普格县| 颍上县| 长海县| 民权县| 南漳县| 三台县| 鹤山市| 垦利县| 吉隆县| 富宁县| 库尔勒市| 措勤县| 喀喇沁旗| 敖汉旗| 新昌县| 潍坊市| 鄂托克旗| 古交市| 固始县| 新安县| 屏东市| 南开区| 常山县| 沙河市| 花莲市| 屏东市| 中方县| 福安市| 开阳县| 安乡县| 屏东市| 高州市| 隆化县|