欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

python裝飾器深入學(xué)習(xí)

 更新時(shí)間:2018年04月06日 14:22:09   作者:嵌動(dòng)初心(aaron)  
這篇文章主要深入學(xué)習(xí)了python裝飾器的相關(guān)資料,什么是裝飾器?裝飾器遵循的原則等,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

什么是裝飾器

在我們的軟件產(chǎn)品升級(jí)時(shí),常常需要給各個(gè)函數(shù)新增功能,而在我們的軟件產(chǎn)品中,相同的函數(shù)可能會(huì)被調(diào)用上百次,這種情況是很常見(jiàn)的,如果我們一個(gè)個(gè)的修改,那我們的碼農(nóng)豈不要掛掉了(有人就說(shuō)了 ,你笨呀,修改函數(shù)定義不就行了!同學(xué),你醒醒吧,如果要新加的功能會(huì)修改參數(shù),或者返回值呢?)。這個(gè)時(shí)候,就是我們裝飾器大顯神通的時(shí)候了。裝飾器就可以實(shí)現(xiàn),在不改變?cè)瘮?shù)的調(diào)用形式下(即函數(shù)的透明化處理),給函數(shù)新增功能的作用。如何實(shí)現(xiàn),以及實(shí)現(xiàn)原理,下文會(huì)詳解。

裝飾器遵循的原則

裝飾器,顧名思義就是起裝飾的作用,既然是裝飾,那么被裝飾的對(duì)象是啥樣就是啥樣,不能有絲毫改變。在這里,我們寫(xiě)裝飾器就是必須把握不能修改被修飾函數(shù)的源代碼這條鐵律。如何遵循這條鐵律,我們還需還需做一些鋪墊,必須先要了解三個(gè)概念,如下:

函數(shù)名即“變量”

在python中,函數(shù)名其實(shí)就像是c語(yǔ)言的函數(shù)指針,代表的是我們的函數(shù)地址,只有解釋器獲取到這個(gè)地址,它才會(huì)去執(zhí)行這塊內(nèi)存的代碼。因此,本質(zhì)上,函數(shù)名就和不同變量沒(méi)什么區(qū)別,只不過(guò)函數(shù)名和普通變量所指代的那塊內(nèi)存的使用方式不同罷了,這些都是底層解釋器的機(jī)制所決定的,對(duì)于程序猿來(lái)說(shuō),都是透明的,所以,我們可以認(rèn)為兩者是沒(méi)有區(qū)別的。

高階函數(shù)

什么是高階函數(shù)其實(shí)很簡(jiǎn)單,把握兩個(gè)原則就好:

  • 形式參數(shù)有函數(shù)名
  • 返回值有函數(shù)名

只要滿(mǎn)足這兩個(gè)原則之一,就可以稱(chēng)之為是高階函數(shù)。翻回頭來(lái)看,這里出現(xiàn)了我們上面說(shuō)的函數(shù)名,仔細(xì)體會(huì)一下,我們?cè)谶@里不就是把其當(dāng)成實(shí)參看待的嗎?

嵌套函數(shù)

什么是嵌套函數(shù)其實(shí)也非常簡(jiǎn)單,把握一個(gè)原則就好:

  • 在一個(gè)函數(shù)的函數(shù)體中去定義另一個(gè)函數(shù)

在這里需要強(qiáng)調(diào)的是,函數(shù)定義時(shí)是不會(huì)執(zhí)行函數(shù)體的,就和定義變量是不會(huì)去讀取變量里的內(nèi)容一樣。這一點(diǎn)至關(guān)重要,對(duì)于我們理解裝飾器實(shí)現(xiàn)原理非常有幫助。

如何寫(xiě)裝飾器

有了上文的鋪墊,在現(xiàn)在來(lái)詳解一下如何寫(xiě)裝飾器,就好理解多了。

裝飾器本質(zhì)

  其實(shí)裝飾器本質(zhì)上就是一個(gè)函數(shù),它也具有函數(shù)名,參數(shù)和返回值。但在python中,我們用“@auth”來(lái)表示。

@auth # 其等價(jià)于:func = auth(func)
def func():
 print("func called")

 這個(gè)示例就是python中如何修飾func函數(shù)的格式,當(dāng)然我們還沒(méi)有實(shí)現(xiàn)我們的裝飾器函數(shù)。我們要注意的是注釋里寫(xiě)的內(nèi)容,我們可以看出:

  • 裝飾器函數(shù)其實(shí)是一個(gè)高階函數(shù)(參數(shù)和返回值都為函數(shù)名)。
  • “auth(func)”是在調(diào)用我們的裝飾器函數(shù),即裝飾器函數(shù)的函數(shù)體會(huì)被執(zhí)行,一定要記好這一點(diǎn)。

設(shè)計(jì)思路

裝飾器即然是個(gè)函數(shù),又有上述介紹的等價(jià)關(guān)系,那我們就可以這樣設(shè)計(jì)我們的裝飾器:

  • 在我們裝飾器的函數(shù)體內(nèi)去定義一個(gè)新的函數(shù),在這個(gè)新定義的函數(shù)內(nèi)去調(diào)用被修飾的函數(shù),與此同時(shí),在被修飾的函數(shù)的上下文去添加新功能。最后,利用裝飾器函數(shù)的返回值返回我們新定義函數(shù)的函數(shù)名。
  • 由此可以知道,“func = auth(func)”中的返回值func表示的就是在裝飾器中新定義的函數(shù)的函數(shù)名。

前面做了大量的鋪墊,就是想在這里揭示裝飾器的實(shí)現(xiàn)機(jī)制,其實(shí)沒(méi)什么什么的,很簡(jiǎn)單:

  • 裝飾器機(jī)制改變了被修飾函數(shù)的函數(shù)名表示的地址數(shù)據(jù)。說(shuō)白了就是,被修飾前,函數(shù)名代表的是A內(nèi)存塊;被修飾后,函數(shù)名代表的是B內(nèi)存塊;只不過(guò),在執(zhí)行B內(nèi)存塊時(shí),會(huì)調(diào)用A內(nèi)存塊罷了。B內(nèi)存塊中的代碼就是我們新加的功能。而這種機(jī)制的實(shí)現(xiàn),使用了“高階函數(shù)”和“嵌套函數(shù)”的機(jī)制。
  • 最終的效果就是,但在調(diào)用被修飾過(guò)的函數(shù)時(shí),其實(shí)調(diào)用的不是原來(lái)的內(nèi)存塊,而是修飾器新申請(qǐng)的內(nèi)存塊。

第一步:設(shè)計(jì)裝飾器函數(shù)

裝飾器函數(shù)定義跟普通函數(shù)定義沒(méi)什么區(qū)別,關(guān)鍵是函數(shù)體怎么寫(xiě)的問(wèn)題。這里,為了便于理解,先用無(wú)參數(shù)的裝飾器函數(shù)說(shuō)明。

#裝飾器函數(shù)定義格式
def deco(func):
 '''函數(shù)體...'''
return func

這里說(shuō)的無(wú)參數(shù),指的是沒(méi)有除了“func”之外的參數(shù)
難點(diǎn)是函數(shù)體的編寫(xiě),下面的示例先告訴你為什么要有第二步:

#使用語(yǔ)法糖@來(lái)裝飾函數(shù),相當(dāng)于“myfunc = deco(myfunc)”
def deco(func):
 print("before myfunc() called.")
 func()
 print("after myfunc() called.")
 return func
 
@deco
def myfunc():
 print("myfunc() called.")
 
myfunc()
myfunc()
 
#output:
before myfunc() called.
myfunc() called.
after myfunc() called.
myfunc() called.
myfunc() called. 

由輸出結(jié)果可以看出,我們的裝飾器并沒(méi)有生效。別跟我說(shuō)裝飾器只生效了一次,那是大家忽略了“@deco”的等效機(jī)制。解釋到“@deco”時(shí),會(huì)解釋成“myfunc = deco(myfunc)”。注意了,前面我提到了,這里其實(shí)在調(diào)用deco函數(shù)的,因此,deco的函數(shù)體會(huì)被執(zhí)行。所以output的前三行并不是調(diào)用myfunc函數(shù)時(shí)產(chǎn)生的效果,那有怎能說(shuō)裝飾器生效了一次呢?第二步就是解決裝飾器沒(méi)生效的問(wèn)題的。

第二步:包裝被修飾函數(shù)

#基本格式
def deco(func):
 def _deco()
 #新增功能
 #...
 #...
 func() #別修飾函數(shù)調(diào)用
 return_deco

 下面給出個(gè)示例:

#使用內(nèi)嵌包裝函數(shù)來(lái)確保每次新函數(shù)都被調(diào)用,
#內(nèi)嵌包裝函數(shù)的形參和返回值與原函數(shù)相同,裝飾函數(shù)返回內(nèi)嵌包裝函數(shù)對(duì)象
 
def deco(func):
 def _deco():
 print("before myfunc() called.")
 func()
 print("after myfunc() called.")
 # 不需要返回func,實(shí)際上應(yīng)返回原函數(shù)的返回值
 return _deco
 
@deco
def myfunc():
 print("myfunc() called.")
 return 'ok'
 
myfunc()
 
#output:
before myfunc() called.
myfunc() called.
after myfunc() called.

  第三步:被修飾函數(shù)參數(shù)和返回值透明化處理

當(dāng)完成了第二步時(shí),其實(shí)裝飾器已經(jīng)完成了主要部分,下面就是對(duì)被修飾函數(shù)的參數(shù)和返回值的處理。這樣才能真正實(shí)現(xiàn)裝飾器的鐵律。話(huà)不多說(shuō),直接上代碼:

#基本格式
def deco(func):
 def _deco(*args, **kwargs) #參數(shù)透明化
 #新增功能
 #...
 #...
 res = func(*args, **kwargs) #別修飾函數(shù)調(diào)用
 return res #返回值透明化
 return_deco

通過(guò)上面的分析知:

參數(shù)透明化:當(dāng)我們?cè)谡{(diào)用被裝飾后的函數(shù)時(shí),其實(shí)調(diào)用的時(shí)這里的_deco函數(shù)。那么,我們就給_deco函數(shù)加上可變參數(shù),并把得到的可變參數(shù)傳遞給func函數(shù)不就可以了。
返回值透明化:和參數(shù)透明化同理,給_deco函數(shù)定義返回值,并返回func的返回值就可以了。

透明化處理就是這么簡(jiǎn)單!至此,我們的裝飾器編寫(xiě)完成。給個(gè)示例吧:

#對(duì)帶參數(shù)的函數(shù)進(jìn)行裝飾,
#內(nèi)嵌包裝函數(shù)的形參和返回值與原函數(shù)相同,裝飾函數(shù)返回內(nèi)嵌包裝函數(shù)對(duì)象
 
def deco(func):
 def _deco(*agrs, **kwagrs):
 print("before myfunc() called.")
 ret = func(*agrs, **kwagrs)
 print(" after myfunc() called. result: %s" % ret)
 return ret
 return _deco
 
@deco
def myfunc(a, b):
 print(" myfunc(%s,%s) called." % (a, b))
 return a + b
 
print("sum=",myfunc(1, 2))
print("sum=",myfunc(3, 4))
 
#output:
before myfunc() called.
 myfunc(1,2) called.
 after myfunc() called. result: 3
sum= 3
before myfunc() called.
 myfunc(3,4) called.
 after myfunc() called. result: 7
sum= 7

裝飾器進(jìn)階

帶參數(shù)裝飾器

裝飾器即然也是函數(shù),那么我們也可以給其傳遞參數(shù)。我這里說(shuō)的是:“@auth(auth_type = 'type1')”這中形式喲。先上個(gè)代碼吧:

#基本格式
def deco(deco_type)
 def _deco(func):
 def __deco(*args, **kwargs) #參數(shù)透明化
  #新增功能
  #...
  #...
  print("deco_type:",deco_type) #使用裝飾器參數(shù)
  res = func(*args, **kwargs) #別修飾函數(shù)調(diào)用
  return res #返回值透明化
 return __deco
 return_deco 

 說(shuō)白了,就是在原來(lái)的裝飾器的基礎(chǔ)上再在最外層套一個(gè)deco函數(shù),并用其來(lái)接收裝飾器參數(shù)。由于是在最外層套了一個(gè)函數(shù),那么這個(gè)函數(shù)的形參的作用范圍就是函數(shù)體內(nèi)部,所以里面的函數(shù)定義中隨便用啦,就這么任性。
那怎么理解解釋器的解析過(guò)程呢?在這里,只要我們明白一點(diǎn)就好,那就是: “@auth(auth_type = 'type1')”等價(jià)于“func = auth(auth_type = 'type1')(func)” 。解釋器會(huì)先翻譯“auth(auth_type = 'type1')”,再將其返回值(假設(shè)給了_func這個(gè)不存在的函數(shù)名)當(dāng)作函數(shù)指針,這里的_func函數(shù)名代表的是_deco,然后再去執(zhí)行“func = _func(func)”,而這個(gè)func函數(shù)名代表的其實(shí)就是__deco。

至此,就達(dá)到了通過(guò)裝飾器來(lái)傳參的目的。給個(gè)示例吧:

#示例7: 在示例4的基礎(chǔ)上,讓裝飾器帶參數(shù),
#和上一示例相比在外層多了一層包裝。
#裝飾函數(shù)名實(shí)際上應(yīng)更有意義些
 
def deco(deco_type):
 def _deco(func):
 def __deco(*args, **kwagrs):
  print("before %s called [%s]." % (func.__name__, deco_type))
  func(*args, **kwagrs)
  print(" after %s called [%s]." % (func.__name__, deco_type))
 return __deco
 return _deco
 
@deco("mymodule")
def myfunc():
 print(" myfunc() called.")
 
@deco("module2")
def myfunc2():
 print(" myfunc2() called.")
 
myfunc()
myfunc2()
 
#output:
before myfunc called [mymodule].
 myfunc() called.
 after myfunc called [mymodule].
before myfunc2 called [module2].
 myfunc2() called.
 after myfunc2 called [module2].

多重裝飾器修飾函數(shù)

如果說(shuō),我上面說(shuō)的內(nèi)容都理解了,那么這個(gè)東東,就太簡(jiǎn)單不過(guò)了。不就是把我們的是裝飾器當(dāng)中被修飾的函數(shù),對(duì)它進(jìn)行裝飾嗎?但我在這里還想說(shuō)的是,我們換個(gè)角度看問(wèn)題。我們的關(guān)注點(diǎn)放在原來(lái)的被修飾的函數(shù)上,就會(huì)發(fā)現(xiàn),NB呀,我可以給它添加若干個(gè)功能撒。給個(gè)示例吧:

def deco(deco_type):
 def _deco(func):
 def __deco(*args, **kwagrs):
  print("before %s called [%s]." % (func.__name__, deco_type))
  func(*args, **kwagrs)
  print(" after %s called [%s]." % (func.__name__, deco_type))
 return __deco
 return _deco
 
@deco("module1")
@deco("mymodule")
def myfunc():
 print(" myfunc() called.")
 
@deco("module2")
def myfunc2():
 print(" myfunc2() called.")
 
myfunc()
 
#output:
before __deco called [module1].
before myfunc called [mymodule].
 myfunc() called.
 after myfunc called [mymodule].
 after __deco called [module1].

 注意結(jié)果喲,@deco("module1"),來(lái)修飾的deco("mymdule")的,和我們想的是一樣的,完美!

領(lǐng)取干貨:零基礎(chǔ)入門(mén)學(xué)習(xí)python視頻教程

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • python實(shí)現(xiàn)批量處理將圖片粘貼到另一張圖片上并保存

    python實(shí)現(xiàn)批量處理將圖片粘貼到另一張圖片上并保存

    今天小編就為大家分享一篇python實(shí)現(xiàn)批量處理將圖片粘貼到另一張圖片上并保存,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-12-12
  • Python獲取用戶(hù)輸入的方法詳解

    Python獲取用戶(hù)輸入的方法詳解

    在Python中,獲取用戶(hù)輸入是一個(gè)基礎(chǔ)且常見(jiàn)的操作,它允許程序與用戶(hù)進(jìn)行交互,Python通過(guò)內(nèi)置的input()函數(shù)來(lái)實(shí)現(xiàn)這一功能,以下將詳細(xì)探討如何在Python中獲取用戶(hù)輸入,并涵蓋多個(gè)方面,需要的朋友可以參考下
    2024-09-09
  • python調(diào)用百度API實(shí)現(xiàn)人臉識(shí)別

    python調(diào)用百度API實(shí)現(xiàn)人臉識(shí)別

    這篇文章主要介紹了python調(diào)用百度API實(shí)現(xiàn)人臉識(shí)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • Tornado Application的實(shí)現(xiàn)

    Tornado Application的實(shí)現(xiàn)

    本文主要介紹了Tornado Application的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05
  • python中的import語(yǔ)句用法大全

    python中的import語(yǔ)句用法大全

    import語(yǔ)句用來(lái)導(dǎo)入其他python文件(稱(chēng)為模塊module),使用該模塊里定義的類(lèi)、方法或者變量,從而達(dá)到代碼復(fù)用的目的,文中給大家提到import 語(yǔ)句的兩種格式通過(guò)示例代碼介紹的很詳細(xì),需要的朋友參考下吧
    2021-07-07
  • opencv函數(shù)threshold、adaptiveThreshold、Otsu二值化的實(shí)現(xiàn)

    opencv函數(shù)threshold、adaptiveThreshold、Otsu二值化的實(shí)現(xiàn)

    這篇文章主要介紹了opencv函數(shù)threshold、adaptiveThreshold、Otsu二值化的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • Python擴(kuò)展內(nèi)置類(lèi)型詳解

    Python擴(kuò)展內(nèi)置類(lèi)型詳解

    這篇文章主要為大家詳細(xì)介紹了Python擴(kuò)展內(nèi)置類(lèi)型的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-03-03
  • Pytorch技巧:DataLoader的collate_fn參數(shù)使用詳解

    Pytorch技巧:DataLoader的collate_fn參數(shù)使用詳解

    今天小編就為大家分享一篇Pytorch技巧:DataLoader的collate_fn參數(shù)使用詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-01-01
  • python3使用requests模塊爬取頁(yè)面內(nèi)容的實(shí)戰(zhàn)演練

    python3使用requests模塊爬取頁(yè)面內(nèi)容的實(shí)戰(zhàn)演練

    本篇文章主要介紹了python3使用requests模塊爬取頁(yè)面內(nèi)容的實(shí)戰(zhàn)演練,具有一定的參考價(jià)值,有興趣的可以了解一下
    2017-09-09
  • Python?enumerate()計(jì)數(shù)器簡(jiǎn)化循環(huán)

    Python?enumerate()計(jì)數(shù)器簡(jiǎn)化循環(huán)

    這篇文章主要介紹了Python?enumerate()計(jì)數(shù)器簡(jiǎn)化循環(huán),enumerate()最大的優(yōu)點(diǎn)就是它返回一個(gè)帶有計(jì)數(shù)器和值的元組,因此我們不必自己增加計(jì)數(shù)器,下面就來(lái)看看文章具體對(duì)它的詳細(xì)介紹吧,需要的朋友可以參考一下
    2021-12-12

最新評(píng)論