關于你的疑惑,我給你做以下總結,希望對你有幫助
理解Python裝飾器(Decorator)
Python裝飾器看起來類似Java中的注解,然鵝和注解并不相同,不過同樣能夠實現面向切面編程。
想要理解Python中的裝飾器,不得不先理解閉包(closure)這一概念。
閉包看看維基百科中的解釋:
在計算機科學中,閉包(英語:Closure),又稱詞法閉包(LexicalClosure)或函數閉包(functionclosures),是引用了自由變量的函數。這個被引用的自由變量將和這個函數一同存在,即使已經離開了創造它的環境也不例外。
官方的解釋總是不說人話,but--talkischeap,showmethecode:
是一個局部變量,在函數執行之后應該就不會存在了。但是嵌套函數引用了這個變量,將這個局部變量封閉在了嵌套函數中,這樣就形成了一個閉包。
結合這個例子再看維基百科的解釋,就清晰明了多了。閉包就是引用了自有變量的函數,這個函數保存了執行的上下文,可以脫離原本的作用域獨立存在。
下面來看看Python中的裝飾器。
裝飾器一個普通的裝飾器一般是這樣:
這樣就定義了一個打印出方法名及其參數的裝飾器。
調用之:
輸出:
裝飾器在使用時,用了語法,讓人有些困擾。其實,裝飾器只是個方法,與下面的調用方式沒有區別:
語法只是將函數傳入裝飾器函數,并無神奇之處。
值得注意的是,這是python提供的裝飾器。它能把原函數的元信息拷貝到裝飾器里面的func函數中。函數的元信息包括docstring、name、參數列表等等。可以嘗試去除,你會發現的輸出變成了wrapper。
帶參數的裝飾器裝飾器允許傳入參數,一個攜帶了參數的裝飾器將有三層函數,如下所示:
看到這個代碼是不是又有些疑問,內層的decorator函數的參數func是怎么傳進去的?和上面一般的裝飾器不大一樣啊。
其實道理是一樣的,將其語法去除,恢復函數調用的形式一看就明白了:
輸出結果與正常使用裝飾器相同:
至此,裝飾器這個有點費解的特性也沒什么神秘了。
裝飾器這一語法體現了Python中函數是第一公民,函數是對象、是變量,可以作為參數、可以是返回值,非常的靈活與強大。