亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

python裝飾器

系統 1764 0
原文鏈接: https://www.runoob.com/w3cnote/python-func-decorators.html

學習菜鳥教程上一個同學的筆記,寫的很好理解。轉來學習。

原文鏈接:https://www.runoob.com/w3cnote/python-func-decorators.html

開始之前先提醒一下:多個裝飾器的調用順序為:從下往上

每個人都有的內褲主要功能是用來遮羞,但是到了冬天它沒法為我們防風御寒,咋辦?我們想到的一個辦法就是把內褲改造一下,讓它變得更厚更長,這樣一來,它不僅有遮羞功能,還能提供保暖,不過有個問題,這個內褲被我們改造成了長褲后,雖然還有遮羞功能,但本質上它不再是一條真正的內褲了。于是聰明的人們發明長褲,在不影響內褲的前提下,直接把長褲套在了內褲外面,這樣內褲還是內褲,有了長褲后寶寶再也不冷了。裝飾器就像我們這里說的長褲,在不影響內褲作用的前提下,給我們的身子提供了保暖的功效。

談裝飾器前,還要先要明白一件事,Python 中的函數和 Java、C++不太一樣,Python 中的函數可以像普通變量一樣當做參數傳遞給另外一個函數,例如:

            
              def foo():
    print("foo")

def bar(func):
    func()

bar(foo)
            
          

正式回到我們的主題。裝飾器本質上是一個 Python 函數或類,它可以讓其他函數或類在不需要做任何代碼修改的前提下增加額外功能,裝飾器的返回值也是一個函數/類對象。它經常用于有切面需求的場景,比如:插入日志、性能測試、事務處理、緩存、權限校驗等場景,裝飾器是解決這類問題的絕佳設計。有了裝飾器,我們就可以抽離出大量與函數功能本身無關的雷同代碼到裝飾器中并繼續重用。概括的講,裝飾器的作用就是為已經存在的對象添加額外的功能。

先來看一個簡單例子,雖然實際代碼可能比這復雜很多:

            
              def foo():
    print('i am foo')
            
          

現在有一個新的需求,希望可以記錄下函數的執行日志,于是在代碼中添加日志代碼:

            
              def foo():
    print('i am foo')
    logging.info("foo is running")
            
          

如果函數 bar()、bar2() 也有類似的需求,怎么做?再寫一個 logging 在 bar 函數里?這樣就造成大量雷同的代碼,為了減少重復寫代碼,我們可以這樣做,重新定義一個新的函數:專門處理日志 ,日志處理完之后再執行真正的業務代碼

            
              def use_logging(func):
    logging.warn("%s is running" % func.__name__)
    func()

def foo():
    print('i am foo')

use_logging(foo)
            
          

這樣做邏輯上是沒問題的,功能是實現了,但是我們調用的時候不再是調用真正的業務邏輯 foo 函數,而是換成了 use_logging 函數,這就破壞了原有的代碼結構, 現在我們不得不每次都要把原來的那個 foo 函數作為參數傳遞給 use_logging 函數,那么有沒有更好的方式的呢?當然有,答案就是裝飾器。

簡單裝飾器

            
              def use_logging(func):

    def wrapper():
        logging.warn("%s is running" % func.__name__)
        return func()   # 把 foo 當做參數傳遞進來時,執行func()就相當于執行foo()
    return wrapper

def foo():
    print('i am foo')

foo = use_logging(foo)  # 因為裝飾器 use_logging(foo) 返回的時函數對象 wrapper,這條語句相當于  foo = wrapper
foo()                   # 執行foo()就相當于執行 wrapper()
            
          

use_logging 就是一個裝飾器,它一個普通的函數,它把執行真正業務邏輯的函數 func 包裹在其中,看起來像 foo 被 use_logging 裝飾了一樣,use_logging 返回的也是一個函數,這個函數的名字叫 wrapper。在這個例子中,函數進入和退出時 ,被稱為一個橫切面,這種編程方式被稱為面向切面的編程。

@ 語法糖

如果你接觸 Python 有一段時間了的話,想必你對 @ 符號一定不陌生了,沒錯 @ 符號就是裝飾器的語法糖,它放在函數開始定義的地方,這樣就可以省略最后一步再次賦值的操作。

            
              def use_logging(func):

    def wrapper():
        logging.warn("%s is running" % func.__name__)
        return func()
    return wrapper

@use_logging
def foo():
    print("i am foo")

foo()
            
          

如上所示,有了 @ ,我們就可以省去foo = use_logging(foo)這一句了,直接調用 foo() 即可得到想要的結果。你們看到了沒有,foo() 函數不需要做任何修改,只需在定義的地方加上裝飾器,調用的時候還是和以前一樣,如果我們有其他的類似函數,我們可以繼續調用裝飾器來修飾函數,而不用重復修改函數或者增加新的封裝。這樣,我們就提高了程序的可重復利用性,并增加了程序的可讀性。

裝飾器在 Python 使用如此方便都要歸因于 Python 的函數能像普通的對象一樣能作為參數傳遞給其他函數,可以被賦值給其他變量,可以作為返回值,可以被定義在另外一個函數內。

*args、**kwargs

可能有人問,如果我的業務邏輯函數 foo 需要參數怎么辦?比如:

            
              def foo(name):
    print("i am %s" % name)
            
          

我們可以在定義 wrapper 函數的時候指定參數:

            
              def wrapper(name):
        logging.warn("%s is running" % func.__name__)
        return func(name)
    return wrapper
            
          

這樣 foo 函數定義的參數就可以定義在 wrapper 函數中。這時,又有人要問了,如果 foo 函數接收兩個參數呢?三個參數呢?更有甚者,我可能傳很多個。當裝飾器不知道 foo 到底有多少個參數時,我們可以用 *args 來代替:

            
              def wrapper(*args):
        logging.warn("%s is running" % func.__name__)
        return func(*args)
    return wrapper
            
          

如此一來,甭管 foo 定義了多少個參數,我都可以完整地傳遞到 func 中去。這樣就不影響 foo 的業務邏輯了。這時還有讀者會問,如果 foo 函數還定義了一些關鍵字參數呢?比如:

            
              def foo(name, age=None, height=None):
    print("I am %s, age %s, height %s" % (name, age, height))
            
          

這時,你就可以把 wrapper 函數指定關鍵字函數:

            
              def wrapper(*args, **kwargs):
        # args是一個數組,kwargs一個字典
        logging.warn("%s is running" % func.__name__)
        return func(*args, **kwargs)
    return wrapper
            
          

帶參數的裝飾器

裝飾器還有更大的靈活性,例如帶參數的裝飾器,在上面的裝飾器調用中,該裝飾器接收唯一的參數就是執行業務的函數 foo 。裝飾器的語法允許我們在調用時,提供其它參數,比如@decorator(a)。這樣,就為裝飾器的編寫和使用提供了更大的靈活性。比如,我們可以在裝飾器中指定日志的等級,因為不同業務函數可能需要的日志級別是不一樣的。

            
              def use_logging(level):
    def decorator(func):
        def wrapper(*args, **kwargs):
            if level == "warn":
                logging.warn("%s is running" % func.__name__)
            elif level == "info":
                logging.info("%s is running" % func.__name__)
            return func(*args)
        return wrapper

    return decorator

@use_logging(level="warn")
def foo(name='foo'):
    print("i am %s" % name)

foo()
            
          

上面的 use_logging 是允許帶參數的裝飾器。它實際上是對原有裝飾器的一個函數封裝,并返回一個裝飾器。我們可以將它理解為一個含有參數的閉包。當我 們使用@use_logging(level="warn")調用的時候,Python 能夠發現這一層的封裝,并把參數傳遞到裝飾器的環境中。

@use_logging(level="warn")?等價于?@decorator

類裝飾器

沒錯,裝飾器不僅可以是函數,還可以是類,相比函數裝飾器,類裝飾器具有靈活度大、高內聚、封裝性等優點。使用類裝飾器主要依靠類的__call__方法,當使用 @ 形式將裝飾器附加到函數上時,就會調用此方法。

            
              class Foo(object):
    def __init__(self, func):
        self._func = func

    def __call__(self):
        print ('class decorator runing')
        self._func()
        print ('class decorator ending')

@Foo
def bar():
    print ('bar')

bar()
functools.wraps
            
          

使用裝飾器極大地復用了代碼,但是他有一個缺點就是原函數的元信息不見了,比如函數的docstring、__name__、參數列表,先看例子:

            
              # 裝飾器
def logged(func):
    def with_logging(*args, **kwargs):
        print func.__name__      # 輸出 'with_logging'
        print func.__doc__       # 輸出 None
        return func(*args, **kwargs)
    return with_logging

# 函數
@logged
def f(x):
   """does some math"""
   return x + x * x

logged(f)
            
          

不難發現,函數 f 被with_logging取代了,當然它的docstring,__name__就是變成了with_logging函數的信息了。好在我們有functools.wraps,wraps本身也是一個裝飾器,它能把原函數的元信息拷貝到裝飾器里面的 func 函數中,這使得裝飾器里面的 func 函數也有和原函數 foo 一樣的元信息了。

            
              from functools import wraps
def logged(func):
    @wraps(func)
    def with_logging(*args, **kwargs):
        print func.__name__      # 輸出 'f'
        print func.__doc__       # 輸出 'does some math'
        return func(*args, **kwargs)
    return with_logging

@logged
def f(x):
   """does some math"""
   return x + x * x
            
          

裝飾器順序

一個函數還可以同時定義多個裝飾器,比如:

            
              @a
@b
@c
def f ():
    pass
            
          

它的執行順序是從里到外,最先調用最里層的裝飾器,最后調用最外層的裝飾器,它等效于

            
              f = a(b(c(f)))
            
          

?


更多文章、技術交流、商務合作、聯系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: www操操| 一级毛片视频在线 | 精品久久洲久久久久护士 | 日韩视频在线观看一区 | 亚洲一区在线免费 | 欧美日韩顶级毛片www免费看 | 国产精品片 | 91在线视频免费 | 国产伦精品一区二区三区免费迷 | 国产精品91视频 | 三上悠亚在线一区 | 欧美一级在线毛片免费观看 | 欧美成人高清视频 | 色狠狠狠狠综合影视 | 亚洲精品国产字幕久久不卡 | 欧美三级成人理伦 | 亚洲欧美日韩激情在线观看 | 欧美性猛交xxxxbbbb | 亚洲黄色高清 | 天天干在线观看 | 久久久久久久久综合影视网 | 欧美国产精品日韩在线 | 前田香织一区二区中文字幕 | 奇米视频在线观看 | 不卡国产在线 | 五月天婷婷在线播放 | 国产婷婷高清在线观看免费 | 色综合一本到久久亚洲91 | 狠狠色狠狠色综合婷婷tag | 美国一级毛片片aa成人 | 婷婷开心中文字幕 | 亚洲人和日本人jzz护士 | 欧美一级色 | 久热国产视频 | 亚洲精品国产成人99久久 | 一区二区三区乱码 | 亚洲一级色 | 日本不卡一区二区三区 | 91麻豆精品国产91久久久久 | 免费一级a毛片在线播放视 免费一级成人毛片 | 欧美日韩91 |