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

溫馨提示×

溫馨提示×

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

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

Python裝飾器怎么正確使用

發布時間:2023-05-19 17:04:47 來源:億速云 閱讀:117 作者:iii 欄目:編程語言

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

使用環境為: Python 3.6.8

Python裝飾器怎么正確使用

什么是裝飾器

要理解裝飾器之前,我們需要了解什么是閉包函數。

閉包函數

我們簡單寫個demo,再解釋一下什么是閉包函數。

def exterFunc(x):
  def innerFunc(y):
    return x * y
  
  return innerFunc

def main() -> None:
  f = exterFunc(6)
  result = f(5)

  print(result)

if __name__ == '__main__':
  main()

可以看到,上述代碼所示,所謂的閉包函數是指: 閉包函數是指在函數中再定義函數,內部函數可以訪問外部的變量,在外部函數中,將內部函數作為返回值返回。

可以看到上述例子中,我們定義了exterFunc的函數,它將接收一個形參x,在exterFunc函數中又中定義了innerFunc,它也接收一個形參y, 在innerFunc函數中,返回x * y,沒錯,內部函數可以訪問外部函數傳入的變量,最后將exterFunc作為返回值返回,這就是閉包函數。

最簡單的裝飾器

裝飾器是一種很特殊的函數,可以接收函數作為形參,且返回一個新的函數,在我們上一篇介紹生成器的時候,還記得我們使用memory_profiler庫來打印函數的內存運行情況么? 這就是用的裝飾器。

Python裝飾器怎么正確使用

我們可以寫個最簡單的例子,來闡述一下python裝飾器,即:

def foo(func):
    def wrapper():
        print("裝飾器開始運行了")
        func()
        print("裝飾器結束運行了")

    return wrapper

@foo
def sayHello():
    print("hello pdudo in juejin")

def main() -> None:
  sayHello()

if __name__ == '__main__':
  main()

上面代碼,我們定義了一個裝飾器foofoo需要傳入一個函數, foo內部有一個函數wrapper。這樣的函數中包函數,我們將其稱之為閉包函數,后面會介紹閉包函數。言歸正傳,在wrapper函數中,我們可以在運行func函數的時候,再其執行前后語句。

需要調用裝飾器的時候,只需要@加上函數名稱即可。

為什么需要裝飾器

要解釋這個問題,我們可以看來了解下,裝飾器解決了一些什么問題:

  • 解決代碼重復性,對于經常需要實現類似的功能而言,可以將該功能抽離出來,作為裝飾器來調用,從而避免代碼重復。

  • 增強代碼可讀性,在不修改原始代碼的前提下,可以利用裝飾器在函數前后增加代碼,例如 處理異常、記錄日志等等,可以利用裝飾器將附加功能和函數主要功能分開,增加代碼可讀性。

說了那么多,我們來列舉一個最簡單的例子,利用裝飾器打印一下函數的運行時間。

import time

def getExecTimers(func):
  def wrapper():
    startTimes = time.time()
    func()
    endTimes = time.time()
    print("函數運行時間: " , endTimes - startTimes ,"s")
  return wrapper

@getExecTimers
def testFunc():
  print("開始執行函數")
  time.sleep(5)
  print("函數執行結束")

def main() -> None:
  testFunc()
  
if __name__ == '__main__':
  main()

這個裝飾器,會記錄函數的運行時間。可以看到,我們為這個函數增加了一個附屬功能,但是又沒有修改到原始函數。

上述案例,應該可以證明為什么需要使用裝飾器了。

裝飾器用法

上述我們討論了最簡單的裝飾器寫法,并且寫了一個小功能,即打印函數的運行時間。接下來,我們要看下裝飾器的其他寫法。

不是用語法糖調用

還記得上面我們調用裝飾器,是使用的@+裝飾器名稱么? 其實這是python的語法糖,如果不是用語法糖的話,應該是這樣來使用的:

def foo(func):
    def wrapper():
        print("裝飾器開始運行了")
        func()
        print("裝飾器結束運行了")

    return wrapper


def sayHello():
    print("hello pdudo in juejin")

def main() -> None:
  f1 = sayHello
  f2 = foo(f1)

  f2()

if __name__ == '__main__':
  main()

完整的寫法應該如下代碼所示,這是一個完整的閉包調用邏輯。

f1 = sayHello
f2 = foo(f1)

f2()

而在函數前加上@+裝飾器名稱, 是一種python的語法糖

帶參數的裝飾器

這里要做一個鋪墊,在python中,有2個特殊的變量,分別為*args**kwargs,都是用來處理不定量參數的,分別代表的含義為:

  • *args: 將會將參數打包為元組

  • **kwargs: 將會打包字典傳遞給函數

def foo(func):
    def wrapper(*args,**kwargs):
        print("裝飾器開始運行了")
        print("裝飾器捕獲到的參數: " ,args,**kwargs)
        func(*args,**kwargs)
        print("裝飾器結束運行了")

    return wrapper

@foo
def sayHello(a,b,c,dicts):
    print("傳入的參數: " , a,b,c)
    print("傳入的參數: " , dicts)

def main() -> None:
  sayHello(1,2,3,{"name":"juejin"})

if __name__ == '__main__':
  main()

在裝飾器中,若我們要給函數傳遞參數,是需要先將參數傳遞給裝飾器,而在裝飾器中接收后再進行傳遞的,所以代碼才會是這樣的:

def foo(func):
    def wrapper(*args,**kwargs):
        print("裝飾器開始運行了")
        print("裝飾器捕獲到的參數: " ,args,**kwargs)
        func(*args,**kwargs)
        print("裝飾器結束運行了")

首先,我們在做傳遞調用的時候,wrapper應該調用形參來接收,接收后,再進行傳遞給函數func

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

向AI問一下細節

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

AI

大理市| 铜川市| 兴义市| 兰坪| 商南县| 唐山市| 类乌齐县| 东平县| 新泰市| 尉氏县| 徐水县| 惠州市| 无极县| 淮南市| 达州市| 时尚| 福安市| 益阳市| 长顺县| 海原县| 化德县| 牙克石市| 沛县| 内黄县| 新干县| 通州市| 浦东新区| 辽源市| 滦平县| 乌苏市| 开封市| 邹城市| 紫金县| 大名县| 青川县| 二连浩特市| 晋州市| 贵德县| 石台县| 商洛市| 天长市|