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

Python的幾個(gè)高級(jí)語(yǔ)法概念淺析(lambda表達(dá)式閉包裝飾器)

 更新時(shí)間:2016年05月28日 12:44:44   投稿:mdxy-dxy  
本文主要記錄自己對(duì)幾個(gè)高級(jí)語(yǔ)法概念的理解:匿名函數(shù)、lambda表達(dá)式、閉包、裝飾器。這幾個(gè)概念并非Python特有,但本文只限于用Python做說(shuō)明

1. 匿名函數(shù)
匿名函數(shù)(anonymous function)是指未與任何標(biāo)識(shí)符綁定的函數(shù),多用在functional programming languages領(lǐng)域,典型應(yīng)用場(chǎng)合:
1) 作為參數(shù)傳給高階函數(shù)(higher-order function ),如python中的built-in函數(shù)filter/map/reduce都是典型的高階函數(shù)
2) 作為高階函數(shù)的返回值(雖然此處的"值"實(shí)際上是個(gè)函數(shù)對(duì)象)
與命名函數(shù)(named function)相比,若函數(shù)只被調(diào)用1次或有限次,則匿名函數(shù)在語(yǔ)法上更輕量級(jí)。
具體語(yǔ)法上,python通過(guò)lambda語(yǔ)法支持函數(shù)體為表達(dá)式的匿名函數(shù),即:python的lambda表達(dá)式本質(zhì)上是個(gè)匿名函數(shù),但其函數(shù)體只能是個(gè)表達(dá)式,不能包含其它語(yǔ)句。
此外,高級(jí)動(dòng)態(tài)語(yǔ)言常借助匿名函數(shù)實(shí)現(xiàn)閉包(closure)或裝飾器(decorator)等高級(jí)語(yǔ)法。
在一些場(chǎng)合下,lambda表達(dá)式的使用使得python程序看起來(lái)非常簡(jiǎn)潔。例如,下面是根據(jù)value對(duì)dict元素做排序的代碼示例:

>>> foo = {'father' : 65, 'mother' : 62, 'sister' : 38, 'brother' : 29, 'me' : 28}
>>> sorted(foo.iteritems(), key=lambda x: x[1])
[('me', 28), ('brother', 29), ('sister', 38), ('mother', 62), ('father', 65)]

2. 閉包
閉包(closure)本質(zhì)上是一個(gè)包含了其引用環(huán)境(referencing environment)的函數(shù)或函數(shù)引用,這里的"引用環(huán)境"通常由一張表來(lái)維護(hù),該表存儲(chǔ)了函數(shù)體會(huì)訪問(wèn)的非局部變量(non-local variables)的引用。
與C語(yǔ)言中的函數(shù)指針相比,閉包允許嵌套函數(shù)訪問(wèn)其作用域外的non-local變量,這與Python解釋器對(duì)變量的作用域查找規(guī)則有關(guān)(Python支持LEGB的查找規(guī)則,想深究的話,可以參考<Learning Python>第4版第17章Scopes關(guān)于作用域及查找規(guī)則的詳細(xì)講解,或者查看這篇文章 做快速了解)。
對(duì)于運(yùn)行時(shí)內(nèi)存分配模型會(huì)在線性棧上創(chuàng)建局部變量的語(yǔ)言來(lái)說(shuō)(典型如C語(yǔ)言),通常很難支持閉包。因?yàn)檫@些語(yǔ)言底層實(shí)現(xiàn)中,若函數(shù)返回,則函數(shù)中定義的局部變量均會(huì)隨著函數(shù)棧被回收而銷毀。但閉包在底層實(shí)現(xiàn)上要求其要訪問(wèn)的non-local變量在閉包被執(zhí)行的時(shí)候保持有效,直到這個(gè)閉包的生命周期結(jié)束,這意外著這些non-local變量只有在其確定不再被使用時(shí)才能銷毀,而不能隨著定義這些變量的函數(shù)返回銷毀。因此,天生支持閉包的語(yǔ)言通常采用garbage collection的方式管理內(nèi)存,因?yàn)間c機(jī)制保證了變量只有不再被引用時(shí)才會(huì)由系統(tǒng)銷毀并回收其內(nèi)存空間。
具體語(yǔ)法上,閉包通常伴隨著函數(shù)嵌套定義。以Python為例,一個(gè)簡(jiǎn)單的閉包示例如下:

#!/bin/env python
#-*- encoding: utf-8 -*-

def startAt_v1(x):
 def incrementBy(y):
  return x + y 
 print 'id(incrementBy)=%s' % (id(incrementBy))
 return incrementBy

def startAt_v2(x):
 return lambda y: x + y 

if '__main__' == __name__:
 c1 = startAt_v1(2)
 print 'type(c1)=%s, c1(3)=%s' % (type(c1), c1(3))
 print 'id(c1)=%s' % (id(c1))
 
 c2 = startAt_v2(2)
 print 'type(c2)=%s, c2(3)=%s' % (type(c2), c2(3))

執(zhí)行結(jié)果如下:

id(incrementBy)=139730510519782
type(c1)=<type 'function'>, c1(3)=5
id(c1)=139730510519782
type(c2)=<type 'function'>, c2(3)=5

上述示例中,startAt_v1和startAt_v2均實(shí)現(xiàn)了閉包,其中:v1借助嵌套定義函數(shù)實(shí)現(xiàn);v2則借助lambda表達(dá)式/匿名函數(shù)來(lái)實(shí)現(xiàn)。
我們以v1為例對(duì)閉包做說(shuō)明:
1) 函數(shù)startAt_v1接受1個(gè)參數(shù),返回1個(gè)函數(shù)對(duì)象,而這個(gè)函數(shù)對(duì)象的行為由嵌套定義的函數(shù)incrementBy實(shí)現(xiàn)。
2) 對(duì)函數(shù)incrementBy來(lái)說(shuō),變量x就是所謂的non-local變量(因?yàn)閤既非該函數(shù)定義的局部變量,又非普通意義上的全局變量),incrementBy實(shí)現(xiàn)具體的函數(shù)行為并返回。
3) main入口的c1接收到的返回值是個(gè)函數(shù)對(duì)象,從id(incrementBy) == id(c1)可斷定,c1"指向"的對(duì)象與函數(shù)名incrementBy"指向"的其實(shí)是同一個(gè)函數(shù)對(duì)象。
4) 受益于Python對(duì)閉包的支持,與普通函數(shù)的對(duì)象相比,c1指向的對(duì)象可以訪問(wèn)不在其函數(shù)作用域內(nèi)的non-local變量,而這個(gè)變量是由incrementBy的外層包裝函數(shù)startAt_v1的入?yún)⑻峁┑?,于是,相?dāng)于c1指向的函數(shù)對(duì)象對(duì)其外層包裝函數(shù)的入?yún)⒕哂?記憶"功能,通過(guò)調(diào)用外層包裝函數(shù)創(chuàng)建閉包時(shí),不同的入?yún)⒈粌?nèi)層函數(shù)作為引用環(huán)境維護(hù)起來(lái)。
5) 調(diào)用c1(3)時(shí),傳入的參數(shù)與引用環(huán)境維護(hù)的外層包裝函數(shù)的參數(shù)一起運(yùn)算得到最終結(jié)果。
以上步驟分析說(shuō)明了一個(gè)閉包從創(chuàng)建到執(zhí)行的基本原理,理解這個(gè)case后,閉包的概念也應(yīng)該清晰了。

3. 裝飾器
python支持裝飾器(decorator)語(yǔ)法。裝飾器的概念對(duì)于初學(xué)者來(lái)說(shuō)比較晦澀,因?yàn)樗婕暗胶瘮?shù)式編程的幾個(gè)概念(如匿名函數(shù)、閉包),這也是本文先介紹匿名函數(shù)和閉包的原因。

我們引用這篇文章對(duì)裝飾器的定義:
A decorator is a function that takes a function object as an argument, and returns a function object as a return value.
從這個(gè)定義可知,裝飾器本質(zhì)上只是一個(gè)函數(shù),它借助閉包的語(yǔ)法去修改一個(gè)函數(shù)(又稱被裝飾函數(shù))的行為,即decorator其實(shí)是個(gè)閉包函數(shù),該函數(shù)以被裝飾函數(shù)名(這個(gè)函數(shù)名其實(shí)是一個(gè)函數(shù)對(duì)象的引用)作為入?yún)ⅲ陂]包內(nèi)修改被裝飾函數(shù)的行為后,返回一個(gè)新的函數(shù)對(duì)象。
特別說(shuō)明:decorator并非必須以函數(shù)形式出現(xiàn),它可以是任何可被調(diào)用的對(duì)象,例如它也可以class形式出現(xiàn),參見這篇文章給出的例子。
在定義好函數(shù)裝飾器的前提下,當(dāng)外部調(diào)用這個(gè)被裝飾函數(shù)時(shí),decorator的語(yǔ)法糖會(huì)由Python解釋器解釋為先執(zhí)行裝飾器函數(shù),然后在裝飾器返回的新函數(shù)對(duì)象上繼續(xù)執(zhí)行其余語(yǔ)句。
來(lái)個(gè)實(shí)例分析一下:

#!/bin/env python
#-*- encoding: utf-8 -*-

def wrapper(fn):
 def inner(n, m):
  n += 1
  print 'in inner: fn=%s, n=%s, m=%s' % (fn.__name__, n, m)
  return fn(n, m) + 6 // 這里有return且返回值為int對(duì)象
 return inner

@wrapper
def foo(n, m):
 print 'in foo: n=%s, m=%s' % (n, m)
 return n * m

print foo(2, 3)

上面的示例中,foo通過(guò)@wrapper語(yǔ)法糖聲明它的裝飾器是wrapper,在wrapper中,定義了嵌套的inner函數(shù)(該函數(shù)的參數(shù)列表必須與被裝飾函數(shù)foo的參數(shù)列表保持一致),裝飾器wrapper修改foo的行為后,返回inner(注意:由于inner的返回值是個(gè)int對(duì)象,故wrpper最終返回的也是個(gè)int對(duì)象)。
調(diào)用foo(2, 3)時(shí),Python解釋器先調(diào)用wrapper對(duì)foo做行為改寫,然后返回int對(duì)象,不難推測(cè),上述代碼的執(zhí)行結(jié)果如下:

in inner: fn=foo, n=3, m=3
in foo: n=3, m=3
foo(2, 3)=15

相關(guān)文章

  • python利用sklearn包編寫決策樹源代碼

    python利用sklearn包編寫決策樹源代碼

    這篇文章主要為大家詳細(xì)介紹了python利用sklearn包編寫決策樹源代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • 利用python爬取m3u8格式視頻的具體實(shí)現(xiàn)

    利用python爬取m3u8格式視頻的具體實(shí)現(xiàn)

    之前爬取的視頻都是mp4格式的,直接用requests請(qǐng)求就可以直接爬取,最近公司安排了一個(gè)小任務(wù),需要爬取m3u8這種格式的視頻,下面這篇文章主要給大家介紹了關(guān)于利用python爬取m3u8格式視頻的相關(guān)資料,需要的朋友可以參考下
    2022-08-08
  • 使用Python實(shí)現(xiàn)FTP文件自動(dòng)傳輸腳本

    使用Python實(shí)現(xiàn)FTP文件自動(dòng)傳輸腳本

    這篇文章主要為大家詳細(xì)介紹了如何使用Python實(shí)現(xiàn)FTP文件自動(dòng)傳輸腳本,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以了解下
    2023-12-12
  • Jupyter Notebook調(diào)用指定的虛擬環(huán)境的實(shí)現(xiàn)示例

    Jupyter Notebook調(diào)用指定的虛擬環(huán)境的實(shí)現(xiàn)示例

    本文主要介紹了Jupyter Notebook調(diào)用指定的虛擬環(huán)境的實(shí)現(xiàn)示例,,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-07-07
  • 基于Python制作簡(jiǎn)易的windows修改器

    基于Python制作簡(jiǎn)易的windows修改器

    現(xiàn)在應(yīng)該大部分人都使用win11系統(tǒng)吧,win11其實(shí)挺好用哈,只是有一點(diǎn)不好用,就是右鍵的菜單,今天做個(gè)小程序,就是應(yīng)該修改win11的一個(gè)應(yīng)用程序,感興趣的可以了解一下
    2022-08-08
  • 解決python腳本中error: unrecognized arguments: True錯(cuò)誤

    解決python腳本中error: unrecognized arguments: True錯(cuò)誤

    這篇文章主要介紹了解決python腳本中error: unrecognized arguments: True錯(cuò)誤,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-04-04
  • Python繪圖實(shí)現(xiàn)臺(tái)風(fēng)路徑可視化代碼實(shí)例

    Python繪圖實(shí)現(xiàn)臺(tái)風(fēng)路徑可視化代碼實(shí)例

    這篇文章主要介紹了Python繪圖實(shí)現(xiàn)臺(tái)風(fēng)路徑可視化代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-10-10
  • Python編程中Python與GIL互斥鎖關(guān)系作用分析

    Python編程中Python與GIL互斥鎖關(guān)系作用分析

    GIL互斥鎖用來(lái)保護(hù)Python世界里的對(duì)象,防止同一時(shí)刻多個(gè)線程執(zhí)行Python字節(jié)碼,確保線程安全,但也導(dǎo)致Python線程無(wú)法利用多核CPU優(yōu)勢(shì),本文來(lái)探討Python將來(lái)是否有可能去除GIL
    2021-09-09
  • Pytorch轉(zhuǎn)tflite方式

    Pytorch轉(zhuǎn)tflite方式

    這篇文章主要介紹了Pytorch轉(zhuǎn)tflite方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-05-05
  • python 調(diào)用釘釘機(jī)器人的方法

    python 調(diào)用釘釘機(jī)器人的方法

    今天小編就為大家分享一篇python 調(diào)用釘釘機(jī)器人的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-02-02

最新評(píng)論