裝飾器是 Python 的一個(gè)亮點(diǎn),但并不神秘,因?yàn)樗举|(zhì)上就是高階函數(shù)加上閉包,只不過給我們提供了一個(gè)優(yōu)雅的語法糖。 至于為什么要有裝飾器,我覺得有句話說的非常好,裝飾器存在的最大意義就是可以在不改動(dòng)原函數(shù)的代碼和調(diào)用方式的情況下,為函數(shù)增加一些新的功能。
如果不使用裝飾器的話:
打印結(jié)果告訴我們,裝飾器只是類似于 foo=deco(foo) 的一個(gè)語法糖罷了。 至于字節(jié)碼這里就不看了,還是那句話,@ 只是個(gè)語法糖,它和我們直接調(diào)用 foo=deco(foo) 是等價(jià)的,所以理解裝飾器(decorator)的關(guān)鍵就在于理解閉包(closure)。 另外函數(shù)在被裝飾器裝飾之后,整個(gè)函數(shù)其實(shí)就已經(jīng)變了,而為了保留原始信息我們一般會(huì)從 functools 模塊中導(dǎo)入一個(gè) wraps 函數(shù)。當(dāng)然裝飾器還可以寫的更復(fù)雜,比如帶參數(shù)的裝飾器、類裝飾器等等,不過這些都屬于 Python 層級(jí)的東西了,我們就不說了。 另外裝飾器還可以不止一個(gè),如果一個(gè)函數(shù)被多個(gè)裝飾器裝飾,會(huì)有什么表現(xiàn)呢?
解釋器還是從上到下解釋,當(dāng)執(zhí)行到 @deco1 的時(shí)候,肯定要裝飾了,但它下面不是函數(shù),也是一個(gè)裝飾器,于是表示:要不哥們,你先裝飾。然后執(zhí)行 @deco2,但它下面還是一個(gè)裝飾器,于是重復(fù)了剛才的話,把皮球踢給 @deco3。 當(dāng)執(zhí)行 @deco3 的時(shí)候,發(fā)現(xiàn)下面終于是一個(gè)普通的函數(shù)了,于是裝飾了。 deco3 裝飾完畢之后,foo = deco3(foo)。然后 deco2 發(fā)現(xiàn) deco3 已經(jīng)裝飾完畢,那么會(huì)對(duì) deco3 裝飾的結(jié)果再進(jìn)行裝飾,此時(shí) foo = deco2(deco3(foo));同理,再經(jīng)過 deco1 的裝飾,最終得到了 foo = deco1(deco2(deco3(foo)))。 于是最終輸出:
所以當(dāng)有多個(gè)裝飾器的時(shí)候,會(huì)從下往上裝飾;然后執(zhí)行的時(shí)候,會(huì)從上往下執(zhí)行。 以上就是裝飾器相關(guān)的內(nèi)容,可以說非常簡單了,甚至有點(diǎn)水文章的嫌疑,因?yàn)楹诵亩荚谏弦黄榻B的閉包當(dāng)中。還是那句話,理解裝飾器的關(guān)鍵就在于理解閉包。 |
|