淺談python裝飾器探究與參數(shù)的領(lǐng)取
首先上原文:
現(xiàn)在,假設(shè)我們要增強now()函數(shù)的功能,比如,在函數(shù)調(diào)用前后自動打印日志,但又不希望修改now()函數(shù)的定義,這種在代碼運行期間動態(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
>>>
'''
運行結(jié)果在注釋里
尷尬了...g被強制塞了個參數(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的返回值
為了驗證下,我把"G"改成了2
結(jié)果是
TypeError: 'int' object is not callable
OK了,大概意思就是,裝飾器首先被"塞"一個參數(shù),然后,返回值還要被調(diào)用一次,然而好像只有函數(shù)才能被調(dià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'
'''
運行成功.但是....說好的"打印日志功能"呢,#沒錯"這里是G"就是我想要的日志
疑點一: 打印的"這里是G"是第一行,是在輸入"f()"之前發(fā)生的....
疑點二: 輸入"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
機智的我動了歪主意
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)建強大的用戶認(rèn)證系統(tǒng)實例探究
這篇文章主要為大家介紹了Python?Flask-Login構(gòu)建強大的用戶認(rèn)證系統(tǒng)示例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
解決python寫入mysql中datetime類型遇到的問題
今天小編就為大家分享一篇解決python寫入mysql中datetime類型遇到的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-06-06

