91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

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

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

哈巴河县| 博兴县| 九江市| 鹤岗市| 弥渡县| 乌兰县| 汉沽区| 板桥市| 凤庆县| 阳西县| 平湖市| 开鲁县| 元江| 西和县| 建宁县| 连山| 延长县| 江都市| 威宁| 郓城县| 垣曲县| 吉水县| 秭归县| 颍上县| 乐安县| 咸宁市| 富宁县| 简阳市| 新津县| 息烽县| 济源市| 蓬溪县| 神农架林区| 灵山县| 松江区| 重庆市| 古交市| 柳河县| 元朗区| 大庆市| 建昌县|