【先導(dǎo)語】:
1.神奇的裝飾器到底是什么
2.裝飾器的用法和語法糖
3.裝飾器如何添加參數(shù)
裝飾器是python里的一個(gè)非常有意思的部分朴恳,他用于封裝函數(shù)代碼萎羔,顯式的將封裝器應(yīng)用到被封裝的函數(shù)上冗锁,從而使得他們選擇加入到裝飾器指定的功能中溯饵。對(duì)于在函數(shù)運(yùn)行前處理常見前置條件(例如確認(rèn)授權(quán))旗们,或在函數(shù)運(yùn)行后確保清理(輸出清除或異常處理)亲配,裝飾器都非常有用套耕。
【小姐姐】聽不明白谁帕,太繞了!
簡單來說冯袍,裝飾器就是實(shí)現(xiàn)了一個(gè)通用的功能匈挖,然后將這個(gè)通用的功能應(yīng)用到不同的、需要使用這個(gè)功能的函數(shù)上康愤,從而避免每次都在不同函數(shù)上反復(fù)寫相同的功能代碼儡循。
裝飾器的本質(zhì)是一個(gè)函數(shù),他接受被裝飾的函數(shù)作為位置參數(shù)翘瓮,裝飾器通過使用該參數(shù)來執(zhí)行某些操作贮折,然后返回一個(gè)函數(shù)引用裤翩,這個(gè)函數(shù)可以是原始函數(shù)资盅,或者是另外一個(gè)函數(shù)。
我們舉例子說明踊赠,裝飾器是這樣的函數(shù)呵扛,他們接受被裝飾的可調(diào)用函數(shù)作為唯一的參數(shù),并且返回一個(gè)可調(diào)用函數(shù)筐带,
register方法是一個(gè)簡單的裝飾器今穿,它把被裝飾的函數(shù)添加到一個(gè)列表中,然后這里是將未改變的被裝飾函數(shù)返回伦籍,可以看出蓝晒,裝飾器一般是傳入被裝飾函數(shù)的引用,然后經(jīng)過一些指定的處理帖鸦,最后返回值也是一個(gè)函數(shù)引用芝薇。
還有一種更簡單的語法形式:
裝飾器的語法糖:我們這里看到的對(duì)foo進(jìn)行裝飾的方法是運(yùn)用
foo = register(foo)語句,還有一種簡單的用法是在聲明函數(shù)的位置應(yīng)用裝飾器作儿,從而使得代碼更容易閱讀解愤,并且讓人立刻意識(shí)到使用了裝飾器
同時(shí)隶垮,我們?cè)偬釤捯幌逻@里面的幾個(gè)重難點(diǎn):
第一,requires_ints中,decorated這個(gè)變量是內(nèi)嵌作用域的變量健盒,在他調(diào)用退出后,返回的inner函數(shù)是可以記住這個(gè)變量的火脉。
第二牵寺,python不支持函數(shù)的參數(shù)列表的多態(tài),即一個(gè)函數(shù)名只能對(duì)應(yīng)唯一的參數(shù)列表形式妒蛇。
第三机断,在內(nèi)嵌函數(shù)內(nèi)部調(diào)用被裝飾函數(shù)的時(shí)候策添,使用了解包參數(shù),關(guān)于這*args, **kwargs毫缆,的參數(shù)形式唯竹,前面章節(jié)中細(xì)講過。
【小姐姐】那我們也用這個(gè)裝飾器來裝飾一個(gè)函數(shù)苦丁。
這里將名稱foo賦給inner函數(shù)浸颓,而不是賦給原來被定義的函數(shù),如果運(yùn)行foo(3,5)旺拉,將利用傳入的這兩個(gè)參數(shù)運(yùn)行inner函數(shù)产上,inner函數(shù)執(zhí)行類型檢查,然后運(yùn)行被裝飾方法蛾狗,如果傳入的不是整形數(shù)晋涣,例如下面這個(gè)例子,那么裝飾器的附加功能就會(huì)進(jìn)行類型檢查:
其次內(nèi)嵌的函數(shù)和被裝飾的函數(shù)的參數(shù)形式必須完全一樣沉桌,這里用的*args, **kwargs概況函數(shù)參數(shù)的一般形式谢鹊,因此也是完全對(duì)應(yīng)的。
最后說說裝飾器參數(shù)
最后來介紹這個(gè)復(fù)雜一些的話題留凭,裝飾器參數(shù)佃扼。之前我們列舉的常規(guī)例子里,裝飾器只有一個(gè)參數(shù)蔼夜,就是被裝飾的方法兼耀。但是,有時(shí)讓裝飾器自身帶有一些需要的信息求冷,從而使裝飾器可以用恰當(dāng)?shù)姆绞窖b飾方法十分有用瘤运。
這些參數(shù)并不是和被裝飾的函數(shù)并列作為參數(shù)簽名,而是在原有裝飾器的基礎(chǔ)上額外再增加一層封裝匠题,那么拯坟,實(shí)質(zhì)是這個(gè)接受其他參數(shù)的裝飾器并不是一個(gè)實(shí)際的裝飾器,而是一個(gè)返回裝飾器的函數(shù)梧躺。
最終返回的內(nèi)嵌函數(shù)inner是最終使用indent和sort_keys參數(shù)的函數(shù)似谁,這沒有問題
我們?cè)谶@里詳細(xì)解釋說明的是操作順序,看上去我們使用的是@json_output(indent=8)掠哥,作這和之前的裝飾器語法糖看上去有些不同巩踏,實(shí)際上這個(gè)不是最終的裝飾器函數(shù),通過調(diào)用json_output(indent=8)续搀,返回函數(shù)指針actual_decorator塞琼,這個(gè)函數(shù)才是真正放在@后的裝飾器函數(shù),原始的被裝飾函數(shù)最終獲得了內(nèi)涵更豐富的inner函數(shù)對(duì)象禁舷,完成了裝飾過程彪杉,值得一提的是毅往,所謂的裝飾器參數(shù)最終傳給了最內(nèi)層的inner函數(shù)。
記住派近,在定義裝飾器函數(shù)后攀唯,真正的裝飾器函數(shù)只有一個(gè)參數(shù),那就是被裝飾的函數(shù)指針渴丸,而有其他參數(shù)的函數(shù)實(shí)質(zhì)上只是裝飾器的外圍函數(shù)侯嘀,他可以依據(jù)參數(shù)對(duì)裝飾器進(jìn)行進(jìn)一步的定制。一句話:一個(gè)函數(shù)不可能接受被裝飾的方法谱轨,又接受其他參數(shù)