Python必備基礎(chǔ)之閉包和裝飾器知識(shí)總結(jié)
一、閉包
1.1 三要素
必須有一個(gè)內(nèi)嵌函數(shù)
內(nèi)嵌函數(shù)必須引用外部函數(shù)中變量
外部函數(shù)返回值必須是內(nèi)嵌函數(shù)
1.2 語法
# 語法 def 外部函數(shù)名(參數(shù)): 外部變量 def 內(nèi)部函數(shù)名(參數(shù)): 使用外部變量 return 內(nèi)部函數(shù)名 # 調(diào)用 變量 = 外部函數(shù)名(參數(shù)) 變量(參數(shù))
舉個(gè)例子
def func01(): # 外部函數(shù) a = 1 # 外部變量 print('外部變量:',a) def func02(num): #內(nèi)部函數(shù) print("調(diào)用內(nèi)部函數(shù)后:",num + a). # 調(diào)用外部變量 # 調(diào)用 func01() # func02()
像這樣,我們先把func02注釋掉,直接調(diào)用func01是可以調(diào)用成功的,完全沒問題
但是我們?cè)僬{(diào)用func02呢?一定會(huì)報(bào)錯(cuò),為什么?這就涉及到一個(gè)知識(shí)點(diǎn):
函數(shù)內(nèi)部的屬性,都是有生命周期的,都是在函數(shù)執(zhí)行期間
簡(jiǎn)單來說就是func02存在于func01函數(shù)體內(nèi),func01調(diào)用執(zhí)行完
它里面的代碼就執(zhí)行不了,如果想讓它存活執(zhí)行下去,就要return出去
再找一個(gè)變量接收,那么這樣,不管你函數(shù)里怎么樣,我就可以從內(nèi)部使用外部的變量
所以調(diào)用這里不能像上面那么寫:
# 調(diào)用 text = func01() text(3) # 3為參數(shù)
這樣才算整整意義上的閉包
1.3 優(yōu)點(diǎn)
內(nèi)部函數(shù)可以使用外部變量
1.4 缺點(diǎn)
外部變量一直存在于內(nèi)存中,不會(huì)在調(diào)用結(jié)束后釋放,占用內(nèi)存
1.5 作用
實(shí)現(xiàn)python裝飾器
二、裝飾器 Decorator
2.1 定義
在不改變?cè)瘮?shù)的調(diào)用以及內(nèi)部代碼情況下,為其添加新功能的函數(shù)
這個(gè)常見的裝飾器就是你拿到別人的第三方API,假如API接口不允許你修改
但是你覺得他寫的特別low,還需要添加某些功能,那我們就需要使用裝飾器
2.2 語法
def 函數(shù)裝飾器名稱(func): def wrapper(*args, **kwargs): 需要添加的新功能 return func(*args, **kwargs) return wrapper 原函數(shù) = 內(nèi)嵌函數(shù) @函數(shù)裝飾器名稱 def 原函數(shù)名稱(參數(shù)): 函數(shù)體 原函數(shù)(參數(shù))
2.3 本質(zhì)
使用“@函數(shù)裝飾器名稱”修飾原函數(shù),等同于創(chuàng)建與原函數(shù)名稱相同的變量,關(guān)聯(lián)內(nèi)嵌函數(shù);故調(diào)用原函數(shù)時(shí)執(zhí)行內(nèi)嵌函數(shù)。
原函數(shù)名稱 = 函數(shù)裝飾器名稱(原函數(shù)名稱)
2.4 裝飾器鏈
一個(gè)函數(shù)可以被多個(gè)裝飾器修飾,執(zhí)行順序?yàn)閺慕竭h(yuǎn)。
接下來我們寫一個(gè)裝飾器的小案例,來更加清楚一下裝飾的整個(gè)工作流程
故事情境是這樣的,主角是男人和女人,假設(shè)男人女人都是可以上班的,但是呢有不同
男人只能是好好上班,不能生娃;女人可以好好上班,也可以生娃
當(dāng)然我們別反駁啊,是有的國(guó)家的男的也有生娃的技術(shù),但是我們這里就是按照我們?cè)O(shè)定好的來
那當(dāng)我們調(diào)用 man() 的時(shí)候,打印 好好上班,你不能生娃
調(diào)用 woman() 的時(shí)候,打印 好好上班,你可以生娃
def man(): print("好好上班") def woman(): print("好好上班") man() woman()
那我們緊接著構(gòu)建裝飾器,裝飾器名字無所謂,想怎么定義就怎么定義
# 裝飾器函數(shù)帶參數(shù) def arg_func(sex): def func1(b_func): def func2(): if sex == 'man': print("你不可以生娃") if sex == 'woman': print("你可以生娃") return b_func() return func2 return func1 @arg_func(sex='man') def man(): print("好好上班") @arg_func(sex='woman') def woman(): print("好好上班") man() woman()
這個(gè)生成器大概的過程就是:
arg_func(sex='man'/'woman')()() > func1 func1() > func func() > print("你不可以生娃") or print("你可以生娃") > b_func # 然后判斷sex的值,最后return出去,拿到結(jié)果
我們看這個(gè)生成器啊,因?yàn)樗@個(gè)函數(shù)這里**def arg_func(sex)😗*這里是默認(rèn)接收一個(gè)函數(shù)名作為參數(shù)進(jìn)來,但是現(xiàn)在參數(shù)有了,但是函數(shù)名不見了,怎么辦,我們只能是去在這個(gè)函數(shù)里再次寫一個(gè)函數(shù),再傳入函數(shù)名
這個(gè)就是相當(dāng)于只要是有傳遞參數(shù)的話,就要在寫一個(gè)函數(shù)套進(jìn)去,因?yàn)槟氵€有一個(gè)函數(shù)名要進(jìn)行傳參
接下來我們看一下被裝飾的函數(shù)帶參數(shù)
def func1(func): def func2(x, y): print(x, y) x += 5 y += 5 return func(x, y) return func2 @func1 def num_sum(a, b): print(a + b) num_sum(1, 2)
這種裝飾器跟之前的相比,直觀的感受來說代碼減少,更加精簡(jiǎn),所以我們常用的也是這個(gè)較多
它總體來說流程變化不大,就是對(duì)于傳參的形式進(jìn)行了變化,即采用函數(shù)最內(nèi)部傳參
到此這篇關(guān)于Python必備基礎(chǔ)之閉包和裝飾器知識(shí)總結(jié)的文章就介紹到這了,更多相關(guān)Python閉包和裝飾器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python 新建文件夾與復(fù)制文件夾內(nèi)所有內(nèi)容的方法
今天小編就為大家分享一篇Python 新建文件夾與復(fù)制文件夾內(nèi)所有內(nèi)容的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-10-10在Python的Django框架中創(chuàng)建和使用模版
這篇文章主要介紹了在Python的Django框架中創(chuàng)建和使用模版的方法,包括使用manage.py shell來幫助設(shè)置模版的方法,需要的朋友可以參考下2015-07-07自適應(yīng)線性神經(jīng)網(wǎng)絡(luò)Adaline的python實(shí)現(xiàn)詳解
這篇文章主要介紹了自適應(yīng)線性神經(jīng)網(wǎng)絡(luò)Adaline的python實(shí)現(xiàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09Python實(shí)現(xiàn)將內(nèi)容寫入文件的五種方法總結(jié)
本篇帶你詳細(xì)看一下python將內(nèi)容寫入文件的方法以及細(xì)節(jié),主要包括write()方法、writelines()?方法、print()?函數(shù)、使用?csv?模塊、使用?json?模塊,需要的可以參考一下2023-04-04用Python進(jìn)行基礎(chǔ)的函數(shù)式編程的教程
這篇文章主要介紹了用Python進(jìn)行基礎(chǔ)的函數(shù)式編程的教程,除了面向?qū)ο缶幊桃馔?、Python還可以進(jìn)行簡(jiǎn)單的不依賴外部變量的函數(shù)式編程,本文介紹了其中的一些基礎(chǔ),需要的朋友可以參考下2015-03-03Python綜合應(yīng)用名片管理系統(tǒng)案例詳解
這篇文章主要介紹了Python綜合應(yīng)用名片管理系統(tǒng),結(jié)合具體案例形式詳細(xì)分析了Python名片管理系統(tǒng)相關(guān)步驟、原理、實(shí)現(xiàn)方法與操作注意事項(xiàng),需要的朋友可以參考下2020-01-01python實(shí)現(xiàn)NB-IoT模塊遠(yuǎn)程控制
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)NB-IoT模塊遠(yuǎn)程控制,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06Keras自定義實(shí)現(xiàn)帶masking的meanpooling層方式
這篇文章主要介紹了Keras自定義實(shí)現(xiàn)帶masking的meanpooling層方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-06-06