在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

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

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

如何寫(xiě)一個(gè)簡(jiǎn)單的裝飾器

科技綠洲 ? 來(lái)源:Python實(shí)用寶典 ? 作者:Python實(shí)用寶典 ? 2023-11-01 09:54 ? 次閱讀

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

這個(gè)庫(kù)可以幫你做什么呢 ?

其實(shí)很簡(jiǎn)單,就是可以幫你更方便地寫(xiě)python裝飾器代碼,更重要的是,它讓 Python 中被裝飾器裝飾后的方法長(zhǎng)得更像裝飾前的方法。

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

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

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

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

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()

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

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

有一些比較聰明的同學(xué),會(huì)利用 PyCharm 來(lái)自動(dòng)生成裝飾器模板

圖片

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

圖片

2. 使用神庫(kù)

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

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

$ python3 -m pip install decorator

從庫(kù)的名稱(chēng)不難看出,這是一個(gè)專(zhuān)門(mén)用來(lái)解決裝飾器問(wèn)題的第三方庫(kù)。

有了它之后,你會(huì)驚奇的發(fā)現(xiàn),以后自己定義的裝飾器,就再也不需要寫(xiě)嵌套的函數(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ù),第一個(gè)參數(shù)是固定的,都是指被裝飾函數(shù),而后面的參數(shù)都固定使用 可變參數(shù) *args**kw 的寫(xiě)法,代碼被裝飾函數(shù)的原參數(shù)。

這種寫(xiě)法,不得不說(shuō),更加符合直覺(jué),代碼的邏輯也更容易理解。

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

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

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

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

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

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

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

下面是一個(gè)官方的示例

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ù)的第一個(gè)參數(shù),還是被裝飾器 func ,這個(gè)跟之前一樣
  • 而第二個(gè)參數(shù) timelimit 寫(xiě)成了位置參數(shù)的寫(xiě)法,并且有默認(rèn)值
  • 再往后,就還是跟原來(lái)一樣使用了可變參數(shù)的寫(xiě)法

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

4. 簽名問(wèn)題有解決?

我們?cè)谧约簩?xiě)裝飾器的時(shí)候,通常都會(huì)順手加上一個(gè)叫 functools.wraps 的裝飾器,我想你應(yīng)該也經(jīng)常見(jiàn)過(guò),那他有啥用呢?

先來(lái)看一個(gè)例子

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

@wrapper
def wrapped():
    pass

print(wrapped.__name__)
#inner_function

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

這也不難理解,因?yàn)樯线厛?zhí)行func 和下邊 decorator(func) 是等價(jià)的,所以上面 func.__name__ 是等價(jià)于下面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)一個(gè)函數(shù)被裝飾器裝飾過(guò)后,它的簽名信息會(huì)發(fā)生變化(譬如上面看到的函數(shù)名)

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

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

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

from functools import wraps

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

@wrapper
def wrapped():
    pass

print(wrapped.__name__)
# wrapped

那么問(wèn)題就來(lái)了,我們使用了 decorator 之后,是否還會(huì)存在這種簽名的問(wèn)題呢?

寫(xiě)個(gè)例子來(lái)驗(yàn)證一下就知道啦

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,說(shuō)明 decorator 已經(jīng)默認(rèn)幫我們處理了一切可預(yù)見(jiàn)的問(wèn)題。

5. 總結(jié)一下

decorator 是一個(gè)提高裝飾器編碼效率的第三方庫(kù),它適用于對(duì)裝飾器原理感到困惑的新手,可以讓你很輕易的寫(xiě)出更符合人類(lèi)直覺(jué)的代碼。

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

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

這么棒的一個(gè)庫(kù),推薦你使用起來(lái)。

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

    關(guān)注

    3

    文章

    4372

    瀏覽量

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

    關(guān)注

    30

    文章

    4891

    瀏覽量

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

    關(guān)注

    0

    文章

    143

    瀏覽量

    10832
  • python
    +關(guān)注

    關(guān)注

    56

    文章

    4825

    瀏覽量

    86295
收藏 人收藏

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    如何寫(xiě)個(gè)Qt版本的Helloworld程序?

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

    如何寫(xiě)個(gè)簡(jiǎn)易的printf函數(shù)?

    如何寫(xiě)個(gè)簡(jiǎn)易的printf函數(shù)?
    發(fā)表于 04-28 06:47

    如何寫(xiě)個(gè)中斷函數(shù)

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

    如何寫(xiě)個(gè)簡(jiǎn)單的字符設(shè)備驅(qū)動(dòng)程序?

    如何寫(xiě)個(gè)簡(jiǎn)單的字符設(shè)備驅(qū)動(dòng)程序?
    發(fā)表于 12-23 06:26

    設(shè)備驅(qū)動(dòng)的入門(mén)教程 (教你如何寫(xiě)驅(qū)動(dòng))

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

    VB寫(xiě)個(gè)簡(jiǎn)單的音頻播放

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

    module 概述以及如何寫(xiě)個(gè)module

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

    如何寫(xiě)個(gè)簡(jiǎn)短的Python代碼做一個(gè)換臉程序的詳細(xì)概述

    在這篇文章中將介紹如何寫(xiě)個(gè)簡(jiǎn)短(200行)的 Python 腳本,來(lái)自動(dòng)地將幅圖片的臉替換為另幅圖片的臉。
    的頭像 發(fā)表于 07-09 10:48 ?4762次閱讀

    如何寫(xiě)個(gè)會(huì)講笑話的Python程序

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

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

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

    [干貨](méi)手把手教你寫(xiě)個(gè)安卓app

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

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

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

    Python自制簡(jiǎn)單實(shí)用的日志裝飾

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

    Python 自制簡(jiǎn)單實(shí)用的日志裝飾

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

    如何寫(xiě)個(gè)內(nèi)存泄漏檢測(cè)工具

    如何確定有內(nèi)存泄露問(wèn)題,如何定位到內(nèi)存泄露位置,如何寫(xiě)個(gè)內(nèi)存泄漏檢測(cè)工具? 1:概述 內(nèi)存泄露本質(zhì):其實(shí)就是申請(qǐng)調(diào)用malloc/new,但是釋放調(diào)用free/delete有遺漏,或者重復(fù)釋放
    的頭像 發(fā)表于 11-11 16:19 ?1049次閱讀
    主站蜘蛛池模板: 嫩草影院永久入口在线观看 | 在线观看一级毛片 | 日本xxxxxxxxx18| 3p高h文| 欧美1314www伊人久久香网 | 傲视影院午夜毛片 | 四虎成人在线视频 | 美女拍拍拍爽爽爽爽爽爽 | 在线看你懂得 | 广东毛片| 国产色啪午夜免费视频 | 五月天狠狠 | 午夜噜噜噜私人影院在线播放 | 国产免费色视频 | 88av在线视频 | 日本一区二区三区在线观看视频 | 伊人网综合在线视频 | 三级毛片在线免费观看 | 亚洲成人www | 亚洲精品第三页 | 天天噜天天干 | 天天干天天射天天舔 | 欧美刺激午夜性久久久久久久 | 欧美一级欧美三级在线观看 | 久久久久国产精品免费网站 | 在线播放真实国产乱子伦 | 五月亭亭六月丁香 | 免费在线视频播放 | xxx性xxx| 操狠狠| 中年艳妇乱小玩 | 视频二区中文字幕 | 狠狠婷婷 | 真人午夜a一级毛片 | 97se狠狠狠狠狼亚洲综合网 | 久久青草精品一区二区三区 | 久久免费视频精品 | 电影天堂bt | 欧美精品啪啪 | 三级毛片在线播放 | 天天摸天天干天天操 |