Python3_裝飾器

1、裝飾器的定義

裝飾器本質(zhì)上就是一個(gè)函數(shù)苍在,功能是為其他函數(shù)添加功能

裝飾器的原則:

  • 不修改被裝飾函數(shù)的源代碼
  • 不修改被裝飾函數(shù)的調(diào)用方式

2绝页、裝飾器的知識(shí)儲(chǔ)備

先聲明一點(diǎn):裝飾器=高階函數(shù)+函數(shù)嵌套+閉包

2.1荠商、高階函數(shù)

定義:

  • 函數(shù)的參數(shù)是一個(gè)函數(shù)
  • 函數(shù)的返回值是一個(gè)函數(shù)

2.2、嵌套函數(shù)

定義:

  • 在一個(gè)函數(shù)的內(nèi)部又定義一個(gè)函數(shù)

注:在嵌套函數(shù)中外部的函數(shù)稱為外函數(shù)续誉,在函數(shù)內(nèi)部定義的函數(shù)稱為內(nèi)函數(shù)

2.3莱没、閉包

定義:

  • 內(nèi)函數(shù)調(diào)用外函數(shù)的臨時(shí)變量
  • 外函數(shù)的返回值是內(nèi)函數(shù)的引用

代碼示例:

def test(a, b):
    def test2():
        print(a + b)
    return test2

res = test(1, 2)
res()  # 相當(dāng)于調(diào)用test2()

3、裝飾器

3.1酷鸦、裝飾器的基本語(yǔ)法格式

通過(guò)上面已經(jīng)知道了 裝飾器=高階函數(shù)+函數(shù)嵌套+閉包郊愧,下面來(lái)實(shí)現(xiàn)一下裝飾器

使用裝飾器統(tǒng)計(jì)函數(shù)test運(yùn)行的時(shí)間

import time


def timer(func):
    '計(jì)算程序運(yùn)行時(shí)間的函數(shù)'
    def count():
        start_time = time.time()
        func()
        end_time = time.time()
        print("運(yùn)行的時(shí)間是%s" % (end_time - start_time))
    return count


@timer      # @timer就是相當(dāng)與執(zhí)行了timer(test)()
def test():
    time.sleep(3)
    print("函數(shù)test運(yùn)行結(jié)束")

運(yùn)行時(shí)間:

函數(shù)test運(yùn)行結(jié)束

運(yùn)行的時(shí)間是3.0005695819854736

3.2、函數(shù)加上返回值

當(dāng)我們?cè)趖est函數(shù)中加上一個(gè)return返回值的時(shí)候會(huì)發(fā)現(xiàn)在調(diào)用test函數(shù)的時(shí)候并沒(méi)有返回值井佑,原因是函數(shù)test是在裝飾器函數(shù)的內(nèi)函數(shù)中運(yùn)行的,在內(nèi)函數(shù)中并沒(méi)有return值眠寿,修改如下:

import time
def timer(func):
    def count():
        start_time = time.time()
        res = func()
        print(res)
        end_time = time.time()
        print("運(yùn)行的時(shí)間是%s" %(end_time - start_time))
    return count

@timer
def test():
    time.sleep(3)
    print("函數(shù)test運(yùn)行結(jié)束")
    return "這是test的返回值"

test()

3.3躬翁、函數(shù)加上參數(shù)

當(dāng)對(duì)test函數(shù)加上參數(shù)的時(shí)候會(huì)發(fā)現(xiàn)代碼報(bào)錯(cuò),原因是在裝飾器的內(nèi)函數(shù)中我們并沒(méi)有給定形參盯拱,導(dǎo)致test在內(nèi)函數(shù)中運(yùn)行時(shí)無(wú)法接收相應(yīng)的實(shí)參盒发;修改如下:

import time
def timer(func):
    def count(*args, **kwargs):
        start_time = time.time()
        res = func(*args, **kwargs)
        print(res)
        end_time = time.time()
        print("運(yùn)行的時(shí)間是%s" %(end_time - start_time))
    return count

@timer
def test(name, age):
    time.sleep(3)
    print("函數(shù)test運(yùn)行結(jié)束")
    print("名字 【%s】,年齡【%s】" %(name, age))
    return "這是test的返回值"

test("xiaoming", 20)

運(yùn)行結(jié)果:

函數(shù)test運(yùn)行結(jié)束

名字 【xiaoming】狡逢,年齡【20】

這是test的返回值

運(yùn)行的時(shí)間是3.0006966590881348

3.4宁舰、給裝飾器函數(shù)加上參數(shù)

如果需要給用戶選擇test函數(shù)是否要統(tǒng)計(jì)運(yùn)行時(shí)間的話就要給裝飾器添加一個(gè)參數(shù),判斷是否要統(tǒng)計(jì)test運(yùn)行的時(shí)間奢浑;可使用閉包

情況1:需要進(jìn)行運(yùn)算時(shí)間的統(tǒng)計(jì)

import time
def auth(para = True):
    def timer(func):
        def count(*args, **kwargs):
            if para == True:
                start_time = time.time()
                res = func(*args, **kwargs)
                print(res)
                end_time = time.time()
                print("運(yùn)行的時(shí)間是%s" %(end_time - start_time))
            else:
                func(*args, **kwargs)
        return count
    return timer

@auth(para=True)
def test(name, age):
    time.sleep(3)
    print("函數(shù)test運(yùn)行結(jié)束")
    print("名字 【%s】蛮艰,年齡【%s】" %(name, age))
    return "這是test的返回值"

test("xiaoming", 20)

運(yùn)行結(jié)果:

函數(shù)test運(yùn)行結(jié)束

名字 【xiaoming】,年齡【20】

這是test的返回值

運(yùn)行的時(shí)間是3.0004258155822754

情況2:不需要進(jìn)行運(yùn)算時(shí)間的統(tǒng)計(jì)

import time
def auth(para = True):
    def timer(func):
        def count(*args, **kwargs):
            if para == True:
                start_time = time.time()
                res = func(*args, **kwargs)
                print(res)
                end_time = time.time()
                print("運(yùn)行的時(shí)間是%s" %(end_time - start_time))
            else:
                func(*args, **kwargs)
        return count
    return timer

@auth(para=False)
def test(name, age):
    time.sleep(3)
    print("函數(shù)test運(yùn)行結(jié)束")
    print("名字 【%s】雀彼,年齡【%s】" %(name, age))
    return "這是test的返回值"

test("xiaoming", 20)

運(yùn)行結(jié)果:

函數(shù)test運(yùn)行結(jié)束

名字 【xiaoming】壤蚜,年齡【20】

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市徊哑,隨后出現(xiàn)的幾起案子袜刷,更是在濱河造成了極大的恐慌,老刑警劉巖莺丑,帶你破解...
    沈念sama閱讀 211,376評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件著蟹,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡梢莽,警方通過(guò)查閱死者的電腦和手機(jī)萧豆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)昏名,“玉大人炕横,你說(shuō)我怎么就攤上這事∑狭#” “怎么了份殿?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,966評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵膜钓,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我卿嘲,道長(zhǎng)颂斜,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,432評(píng)論 1 283
  • 正文 為了忘掉前任拾枣,我火速辦了婚禮沃疮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘梅肤。我一直安慰自己司蔬,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,519評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布姨蝴。 她就那樣靜靜地躺著俊啼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪左医。 梳的紋絲不亂的頭發(fā)上授帕,一...
    開(kāi)封第一講書(shū)人閱讀 49,792評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音浮梢,去河邊找鬼跛十。 笑死,一個(gè)胖子當(dāng)著我的面吹牛秕硝,可吹牛的內(nèi)容都是我干的芥映。 我是一名探鬼主播,決...
    沈念sama閱讀 38,933評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼远豺,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼屏轰!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起憋飞,我...
    開(kāi)封第一講書(shū)人閱讀 37,701評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤霎苗,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后榛做,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體唁盏,經(jīng)...
    沈念sama閱讀 44,143評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,488評(píng)論 2 327
  • 正文 我和宋清朗相戀三年检眯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了厘擂。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,626評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡锰瘸,死狀恐怖刽严,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情避凝,我是刑警寧澤舞萄,帶...
    沈念sama閱讀 34,292評(píng)論 4 329
  • 正文 年R本政府宣布眨补,位于F島的核電站,受9級(jí)特大地震影響倒脓,放射性物質(zhì)發(fā)生泄漏撑螺。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,896評(píng)論 3 313
  • 文/蒙蒙 一崎弃、第九天 我趴在偏房一處隱蔽的房頂上張望甘晤。 院中可真熱鬧,春花似錦饲做、人聲如沸线婚。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)塞弊。三九已至,卻和暖如春缀踪,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背虹脯。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工驴娃, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人循集。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓唇敞,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親咒彤。 傳聞我的和親對(duì)象是個(gè)殘疾皇子疆柔,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,494評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容

  • 1、裝飾器的定義 裝飾器本質(zhì)上就是一個(gè)函數(shù)镶柱,功能是為其他函數(shù)添加功能 裝飾器的原則: 不修改被裝飾函數(shù)的源代碼 不...
    唐_sri閱讀 1,571評(píng)論 0 1
  • 函數(shù) 什么是函數(shù)? 將具有某種功能的代碼放到一起, 構(gòu)成一個(gè)函數(shù).為什么說(shuō)函數(shù)? 因?yàn)樾枰芯恳粋€(gè)問(wèn)題, 函數(shù)可以...
    DragonFangQy閱讀 911評(píng)論 0 2
  • 要點(diǎn): 函數(shù)式編程:注意不是“函數(shù)編程”旷档,多了一個(gè)“式” 模塊:如何使用模塊 面向?qū)ο缶幊蹋好嫦驅(qū)ο蟮母拍睢傩浴?..
    victorsungo閱讀 1,476評(píng)論 0 6
  • 在學(xué)習(xí)Python的過(guò)程中歇拆,我相信有很多人和我一樣鞋屈,對(duì)Python的裝飾器一直覺(jué)得很困惑,我也是困惑了好久故觅,并通過(guò)...
    愚灬墨閱讀 457評(píng)論 1 1
  • 請(qǐng)編寫(xiě)一個(gè)decorator厂庇,能在函數(shù)調(diào)用的前后打印出'begin call'和'end call'的日志。 再思...
    前鋒游弈使閱讀 231評(píng)論 0 0