淺談python裝飾器探究與參數(shù)的領(lǐng)取
首先上原文:
現(xiàn)在,假設(shè)我們要增強(qiáng)now()函數(shù)的功能,比如,在函數(shù)調(diào)用前后自動打印日志,但又不希望修改now()函數(shù)的定義,這種在代碼運(yùn)行期間動態(tài)增加功能的方式,稱之為“裝飾器”(Decorator)。
本質(zhì)上,decorator就是一個返回函數(shù)的高階函數(shù)。
Decorator本質(zhì)是高階函數(shù)?
不信邪的我試了下..
def g(): print("這里是G") return "G" @g def f(): print("這里是F") return 1 ''' -------------------------------------------- line 5, in <module> @g TypeError: g() takes 0 positional arguments but 1 was given >>> '''
運(yùn)行結(jié)果在注釋里
尷尬了...g被強(qiáng)制塞了個參數(shù),那個參數(shù)應(yīng)該是g"修飾的對象"
修改下繼續(xù).....
def g(f): print("這里是G") return "G" @g def f(): print("這里是F") return 1 ''' -------------------------------------------- 這里是G >>> f() Traceback (most recent call last): File "<pyshell#0>", line 1, in <module> f() TypeError: 'str' object is not callable '''
str對象不能被調(diào)用,,,這里的str只有一個,就是g的返回值
為了驗(yàn)證下,我把"G"改成了2
結(jié)果是
TypeError: 'int' object is not callable
OK了,大概意思就是,裝飾器首先被"塞"一個參數(shù),然后,返回值還要被調(diào)用一次,然而好像只有函數(shù)才能被調(diào)用,,所以,為了不報(bào)錯,裝飾器必須返回一個函數(shù),裝飾器必須是高階函數(shù)......
我表示不服( ̄へ ̄),不就是函數(shù)嗎,g的參數(shù)就是函數(shù),,,騷操作上腦ing
def g(f): print("這里是G") return f @g def f(): print("這里是F") return 1 ''' -------------------------------------------- 這里是G >>> f() 這里是F '1' '''
運(yùn)行成功.但是....說好的"打印日志功能"呢,#沒錯"這里是G"就是我想要的日志
疑點(diǎn)一: 打印的"這里是G"是第一行,是在輸入"f()"之前發(fā)生的....
疑點(diǎn)二: 輸入"f()"后,"這里是G"沒打出來.....
#單看g函數(shù),它不算是"高階函數(shù)"
作為成功的例子,它太失敗了┐( ̄ヘ ̄)┌
##好吧,我投降了,負(fù)隅頑抗也不怎么有意思.....
閉包告訴我們一個道理,,,為了保證返回值一定是函數(shù),最好的措施就是"在函數(shù)內(nèi)部,現(xiàn)場造一個函數(shù)然后扔出去"
def g(f): print("這里是G") def h(): print('這里是H') return "H" return h @g def f(): print("這里是F") return 1 ''' -------------------------------------------- 這里是G >>> f() 這里是H 'H' >>> f() 這里是H 'H' >>> '''
f函數(shù)不執(zhí)行了,,是的沒錯,我還多試了一次的
另外,兩條日志只有里面的能用......(下文會解釋的)
看了一下書,h函數(shù)返回f()的話,f函數(shù)就能被執(zhí)行了,,,,,個人感覺裝飾器應(yīng)該叫"劫持器"
def g(f): print("這里是G") def h(): print('這里是H') return f() return h @g def f(): print("這里是F") return 1 ''' -------------------------------------------- 這里是G >>> f() 這里是H 這里是F '1' >>> f <function g.<locals>.h at 0x0000020CBDBB6C80> '''
按書上的思路解釋下
''' @g def f(): pass >>>f() 等價于 >>>g(f) () g函數(shù)執(zhí)行,返回 >>>h () h函數(shù)執(zhí)行(打印日志) >>>f() f執(zhí)行,返回1 >>>1 '''
加上參數(shù),
def g(f): print("這里是G") def h(*args,**kw): print('這里是H') return f(*args,**kw) return h @g def f(*args,**kw): print("這里是F") return "1" ''' >>>f(*args,**kw) 等價于 >>>g(f) (*args,**kw) g函數(shù)執(zhí)行,返回 >>>h (*args,**kw) h函數(shù)執(zhí)行(打印日志) >>>f(*args,**kw) f執(zhí)行,返回1 >>>1 '''
可以看出,(*args,**kw)本改被h函數(shù)拿走,所以,觀察h函數(shù),h把它的參數(shù)原封不動的交給了f
機(jī)智的我動了歪主意
def g(f): print("這里是G") def h():#h沒要求參數(shù) print('這里是H') return f return h @g def f(*args,**kw): print("這里是F") return "1" ''' >>>f()(*args,**kw) 等價于 >>>g(f) ()(*args,**kw) g函數(shù)執(zhí)行,返回 >>>h ()(*args,**kw) h函數(shù)執(zhí)行,h拿的空參數(shù) (打印日志) >>>f(*args,**kw) f執(zhí)行,返回1 >>>1 '''
可是新的問題來了,后面f調(diào)用的時候得多加個空括號,否則
''' >>>f(*args,**kw) 等價于 >>>g(f) (*args,**kw) g函數(shù)執(zhí)行,返回 >>>h (*args,**kw) h函數(shù)執(zhí)行(打印日志)返回f >>>f 這是一個函數(shù)對象 '''
以上告訴我們一個道理"函數(shù)執(zhí)行不執(zhí)行取決于后面有沒有括號"
舉個例子
def m(a): print(a) return m print(m(1)(2)(3)(4)(5)(6)(7)(8)(9)(10)) ''' ----------------------------- 2 4 6 8 10 <function m at 0x000002832BDB10D0>
燒腦時刻
f = a.b.c()()[0]()[d()()[e]]
a模塊的b類的c方法是高階函數(shù),最終返回一個列表,列表里有個函數(shù)
函數(shù)又返回一個字典............
以上這篇淺談python裝飾器探究與參數(shù)的領(lǐng)取就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Python?Flask-Login構(gòu)建強(qiáng)大的用戶認(rèn)證系統(tǒng)實(shí)例探究
這篇文章主要為大家介紹了Python?Flask-Login構(gòu)建強(qiáng)大的用戶認(rèn)證系統(tǒng)示例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01解決python寫入mysql中datetime類型遇到的問題
今天小編就為大家分享一篇解決python寫入mysql中datetime類型遇到的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-06-06