0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何寫一個簡單的裝飾器

科技綠洲 ? 來源:Python實用寶典 ? 作者:Python實用寶典 ? 2023-11-01 09:54 ? 次閱讀

今天介紹的是一個已經(jīng)存在十三年,但是依舊不紅的庫 decorator,好像很少有人知道他的存在一樣。

這個庫可以幫你做什么呢 ?

其實很簡單,就是可以幫你更方便地寫python裝飾器代碼,更重要的是,它讓 Python 中被裝飾器裝飾后的方法長得更像裝飾前的方法。

本篇文章不會過多的向你介紹裝飾器的基本知識,我會默認(rèn)你知道什么是裝飾器,并且懂得如何寫一個簡單的裝飾器。

不了解裝飾器的可以先去閱讀我之前寫的文章,非常全且詳細(xì)的介紹了裝飾器的各種實現(xiàn)方法。

1. 常規(guī)的裝飾器

下面這是一個最簡單的裝飾器示例,在運行 myfunc 函數(shù)的前后都會打印一條日志。

def deco(func):
    def wrapper(*args, **kw):
        print("Ready to run task")
        func(*args, **kw)
        print("Successful to run task")
    return wrapper

@deco
def myfunc():
    print("Running the task")

myfunc()

裝飾器使用起來,似乎有些高端和魔幻,對于一些重復(fù)性的功能,往往我們會封裝成一個裝飾器函數(shù)。

在定義一個裝飾器的時候,我們都需要像上面一樣機械性的寫一個嵌套的函數(shù),對裝飾器原理理解不深的初學(xué)者,往往過段時間就會忘記如何定義裝飾器。

有一些比較聰明的同學(xué),會利用 PyCharm 來自動生成裝飾器模板

圖片

然后要使用的時候,直接敲入 deco 就會生成一個簡單的生成器代碼,提高編碼的準(zhǔn)備效率

圖片

2. 使用神庫

使用 PyCharm 的 Live Template ,雖然能降低編寫裝飾器的難度,但卻要依賴 PyCharm 這一專業(yè)的代碼編輯器。

這里,明哥要教你一個更加簡單的方法,使用這個方法呢,你需要先安裝一個庫 :decorator,使用 pip 可以很輕易地去安裝它

$ python3 -m pip install decorator

從庫的名稱不難看出,這是一個專門用來解決裝飾器問題的第三方庫。

有了它之后,你會驚奇的發(fā)現(xiàn),以后自己定義的裝飾器,就再也不需要寫嵌套的函數(shù)了

from decorator import decorator

@decorator
def deco(func, *args, **kw):
    print("Ready to run task")
    func(*args, **kw)
    print("Successful to run task")

@deco
def myfunc():
    print("Running the task")

myfunc()

deco 作為裝飾函數(shù),第一個參數(shù)是固定的,都是指被裝飾函數(shù),而后面的參數(shù)都固定使用 可變參數(shù) *args**kw 的寫法,代碼被裝飾函數(shù)的原參數(shù)。

這種寫法,不得不說,更加符合直覺,代碼的邏輯也更容易理解。

3. 帶參數(shù)的裝飾器可用?

裝飾器根據(jù)有沒有攜帶參數(shù),可以分為兩種

第一種 :不帶參數(shù),最簡單的示例,上面已經(jīng)舉例

def decorator(func):
    def wrapper(*args, **kw):
        func(*args, **kw)
    return wrapper

第二種 :帶參數(shù),這就相對復(fù)雜了,理解起來了也不是那么容易。

def decorator(arg1, arg2):
    def wrapper(func):
        def deco(*args, **kwargs)
            func(*args, **kwargs)
        return deco
    return wrapper

那么對于需要帶參數(shù)的裝飾器,decorator 是否也一樣能很好的支持呢?

下面是一個官方的示例

from decorator import decorator

@decorator
def warn_slow(func, timelimit=60, *args, **kw):
    t0 = time.time()
    result = func(*args, **kw)
    dt = time.time() - t0
    if dt > timelimit:
        logging.warn('%s took %d seconds', func.__name__, dt)
    else:
        logging.info('%s took %d seconds', func.__name__, dt)
    return result

@warn_slow(timelimit=600)  # warn if it takes more than 10 minutes
def run_calculation(tempdir, outdir):
    pass

可以看到

  • 裝飾函數(shù)的第一個參數(shù),還是被裝飾器 func ,這個跟之前一樣
  • 而第二個參數(shù) timelimit 寫成了位置參數(shù)的寫法,并且有默認(rèn)值
  • 再往后,就還是跟原來一樣使用了可變參數(shù)的寫法

不難推斷,只要你在裝飾函數(shù)中第二個參數(shù)開始,使用了非可變參數(shù)的寫法,這些參數(shù)就可以做為裝飾器調(diào)用時的參數(shù)。

4. 簽名問題有解決?

我們在自己寫裝飾器的時候,通常都會順手加上一個叫 functools.wraps 的裝飾器,我想你應(yīng)該也經(jīng)常見過,那他有啥用呢?

先來看一個例子

def wrapper(func):
    def inner_function():
        pass
    return inner_function

@wrapper
def wrapped():
    pass

print(wrapped.__name__)
#inner_function

為什么會這樣子?不是應(yīng)該返回 func 嗎?

這也不難理解,因為上邊執(zhí)行func 和下邊 decorator(func) 是等價的,所以上面 func.__name__ 是等價于下面decorator(func).__name__ 的,那當(dāng)然名字是 inner_function

def wrapper(func):
    def inner_function():
        pass
    return inner_function

def wrapped():
    pass

print(wrapper(wrapped).__name__)
#inner_function

目前,我們可以看到當(dāng)一個函數(shù)被裝飾器裝飾過后,它的簽名信息會發(fā)生變化(譬如上面看到的函數(shù)名)

那如何避免這種情況的產(chǎn)生?

解決方案就是使用我們前面所說的 functools .wraps 裝飾器。

它的作用就是將 被修飾的函數(shù)(wrapped) 的一些屬性值賦值給 修飾器函數(shù)(wrapper) ,最終讓屬性的顯示更符合我們的直覺。

from functools import wraps

def wrapper(func):
    @wraps(func)
    def inner_function():
        pass
    return inner_function

@wrapper
def wrapped():
    pass

print(wrapped.__name__)
# wrapped

那么問題就來了,我們使用了 decorator 之后,是否還會存在這種簽名的問題呢?

寫個例子來驗證一下就知道啦

from decorator import decorator

@decorator
def deco(func, *args, **kw):
    print("Ready to run task")
    func(*args, **kw)
    print("Successful to run task")

@deco
def myfunc():
    print("Running the task")

print(myfunc.__name__)

輸出的結(jié)果是 myfunc,說明 decorator 已經(jīng)默認(rèn)幫我們處理了一切可預(yù)見的問題。

5. 總結(jié)一下

decorator 是一個提高裝飾器編碼效率的第三方庫,它適用于對裝飾器原理感到困惑的新手,可以讓你很輕易的寫出更符合人類直覺的代碼。

對于帶參數(shù)裝飾器的定義,是非常復(fù)雜的,它需要要寫多層的嵌套函數(shù),并且需要你熟悉各個參數(shù)的傳遞路徑,才能保證你寫出來的裝飾器可以正常使用。

這時候,只要用上 decorator 這個庫,你就可以很輕松的寫出一個帶參數(shù)的裝飾器。同時你也不用擔(dān)心他會出現(xiàn)簽名問題,這些它都為你妥善的處理好了。

這么棒的一個庫,推薦你使用起來。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4277

    瀏覽量

    62323
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4722

    瀏覽量

    68230
  • 日志
    +關(guān)注

    關(guān)注

    0

    文章

    138

    瀏覽量

    10626
  • python
    +關(guān)注

    關(guān)注

    55

    文章

    4767

    瀏覽量

    84375
收藏 人收藏

    評論

    相關(guān)推薦

    如何寫Qt版本的Helloworld程序?

    如何寫Qt版本的Helloworld程序?
    發(fā)表于 05-05 11:07

    如何寫簡易的printf函數(shù)?

    如何寫簡易的printf函數(shù)?
    發(fā)表于 04-28 06:47

    如何寫中斷函數(shù)

    目錄1、中斷2、復(fù)位3、通用I/O及其中斷:I/O寄存I/O中斷寄存默認(rèn)狀態(tài)如何寫中斷函數(shù)總結(jié)所整理歸納僅做分享與記錄,從自身角度分
    發(fā)表于 11-29 07:30

    如何寫簡單的字符設(shè)備驅(qū)動程序?

    如何寫簡單的字符設(shè)備驅(qū)動程序?
    發(fā)表于 12-23 06:26

    設(shè)備驅(qū)動的入門教程 (教你如何寫驅(qū)動)

    設(shè)備驅(qū)動的入門教程 (教你如何寫驅(qū)動)
    發(fā)表于 03-28 09:44 ?61次下載

    VB簡單的音頻播放

    VB簡單的音頻播放 從這里下載整個VB程序包及可執(zhí)行文件
    發(fā)表于 11-09 16:50 ?38次下載

    module 概述以及如何寫module

    driver,而且它也能縮短我們 driver development 的時間。在這篇文章里,我將要跟各位介紹下 module 的原理,以及如何寫 module。 module
    發(fā)表于 11-07 11:37 ?0次下載

    如何寫簡短的Python代碼做一個換臉程序的詳細(xì)概述

    在這篇文章中將介紹如何寫簡短(200行)的 Python 腳本,來自動地將幅圖片的臉替換為另幅圖片的臉。
    的頭像 發(fā)表于 07-09 10:48 ?4475次閱讀

    如何寫會講笑話的Python程序

    笑話從哪里來?自己肯定是不現(xiàn)實的。在這個“云”的時代,各種云都有,自然是不缺開放API的。下面用Python的例子,其實不止pyt
    的頭像 發(fā)表于 01-11 16:47 ?3039次閱讀
    <b class='flag-5'>如何寫</b><b class='flag-5'>一</b><b class='flag-5'>個</b>會講笑話的Python程序

    讓你學(xué)Python裝飾的五大理由

    你必須學(xué)Python裝飾的五理由
    的頭像 發(fā)表于 03-02 10:06 ?1876次閱讀

    [干貨]手把手教你安卓app

    摘要:最近有很多小伙伴在后臺留言:Android Studio。我想大家是想手機app,前面已經(jīng)分享了在QT上如何寫
    發(fā)表于 01-25 18:14 ?7次下載
    [干貨]手把手教你<b class='flag-5'>寫</b><b class='flag-5'>一</b><b class='flag-5'>個</b>安卓app

    如何寫要被C調(diào)用的匯編函數(shù)

    問:我該怎么將被C調(diào)用的匯編函數(shù)? 答:簡單的學(xué)習(xí)怎么
    的頭像 發(fā)表于 01-22 14:54 ?1234次閱讀

    Python自制簡單實用的日志裝飾

    代碼的時候,往往會漏掉日志這個關(guān)鍵因素,導(dǎo)致功能在使用的時候出錯卻無法溯源。 其實,只需要寫非常簡單的日志裝飾
    的頭像 發(fā)表于 10-21 14:39 ?683次閱讀
    Python自制<b class='flag-5'>簡單</b>實用的日志<b class='flag-5'>裝飾</b><b class='flag-5'>器</b>

    Python 自制簡單實用的日志裝飾

    代碼的時候,往往會漏掉日志這個關(guān)鍵因素,導(dǎo)致功能在使用的時候出錯卻無法溯源。 其實,只需要寫非常簡單的日志裝飾
    的頭像 發(fā)表于 10-31 15:05 ?467次閱讀
    Python 自制<b class='flag-5'>簡單</b>實用的日志<b class='flag-5'>裝飾</b><b class='flag-5'>器</b>

    如何寫內(nèi)存泄漏檢測工具

    如何確定有內(nèi)存泄露問題,如何定位到內(nèi)存泄露位置,如何寫內(nèi)存泄漏檢測工具? 1:概述 內(nèi)存泄露本質(zhì):其實就是申請調(diào)用malloc/new,但是釋放調(diào)用free/delete有遺漏,或者重復(fù)釋放
    的頭像 發(fā)表于 11-11 16:19 ?772次閱讀