Python編程:遞歸與匿名函數(shù)及函數(shù)屬性與文檔字符串(函數(shù)補充)
本文簡單扼要地說,輔以代碼進一步地加深理解。
遞歸函數(shù)
當函數(shù)調(diào)用自身而生成最終結(jié)果時,這樣的函數(shù)稱為遞歸。有時遞歸函數(shù)非常有用,因為它們使編寫代碼變得更容易——使用遞歸范式編寫一些算法非常容易,而其他算法則不是這樣。沒有不能以迭代方式重寫的遞歸函數(shù),換句話說,所有遞歸函數(shù)都可以通過循環(huán)迭代的方式實現(xiàn),因此通常由程序員根據(jù)手頭的情況選擇最佳方法。
遞歸函數(shù)主體通常有兩個部分:一部分的返回值依賴于對自身的后續(xù)調(diào)用,另一部分的返回值不依賴于對自身的后續(xù)調(diào)用(稱基本情況,或遞歸邊界)。
作為理解的參考示例,我們看一個階乘函數(shù)N!作為遞歸的兩部分分別是:基本情況(邊界,用來結(jié)束遞歸)是當N為0或1時,函數(shù)返回1,不需要進一步計算。另一方面,在一般情況下的自我調(diào)用,即N!返回的生成結(jié)果:
如果你仔細想想,N!可以寫成這樣:N!= (N - 1) !*N。作為一個實際的例子,請看如下的階乘表示:
我們來轉(zhuǎn)化成函數(shù)實現(xiàn):
高手大俠們在編寫算法時經(jīng)常使用遞歸函數(shù),編寫遞歸函數(shù)非常有趣。作為練習,嘗試使用遞歸和迭代方法解決幾個簡單的問題。很好的練習對象可能是計算斐波那契數(shù)列,或其它諸如此類的東西。自己動手去試試吧。
提示: 在編寫遞歸函數(shù)時,總是考慮要進行多少個嵌套調(diào)用,因為這是有限制的。有關這方面的更多信息,請查看sys.getrecursionlimit()和sys.setrecursionlimit()。 |
匿名函數(shù)
還有一種函數(shù)是匿名函數(shù)(Anonymous functions)。這些函數(shù)在Python中稱為lambda(蘭姆達),其通常在使用具有自己完整定義名稱的函數(shù)有些多余時而使用,此時所需要的只是一個快速、簡單的一行程序來完成這項工作。
假設我們想要一個列表,所有N的某個值,是5的倍數(shù)的數(shù)字。為此,我們可以使用filter()函數(shù),它需要一個函數(shù)和一個可迭代對象作為輸入。返回值是一個過濾器對象,當你遍歷它時,會從輸入可迭代對象中生成元素,所需的參數(shù)函數(shù)會為其返回True。如果不使用匿名函數(shù),我們可能會這樣做:
注意我們?nèi)绾问褂胕sMultipleOfFive()來過濾前n個自然數(shù)。這似乎有點過分——任務及其很簡單,我們不需要為其他任何事情保留isMultipleOfFive()函數(shù)。此時,我們就可用lambda函數(shù)來重寫它:
邏輯是完全相同的,但是過濾函數(shù)現(xiàn)在是個lambda函數(shù),顯然,Lambda更簡單。
定義Lambda函數(shù)非常簡單,它遵循以下形式:
funcName = lambda [parameter_list]: expression
其返回的是一個函數(shù)對象,相當于:
def func_ name([parameter_list]):return expression
參數(shù)列表以逗號分隔。
注意,可選參數(shù)是方括號括起來的部分,是通用語法的表示形式,即文中的方括號部分是可選的,根據(jù)實際需要提供,
我們再來看另外兩個等價函數(shù)的例子,以兩種形式定義:
前面的例子非常簡單。第一個函數(shù)將兩個數(shù)字相加,第二個函數(shù)生成字符串的大寫版本。注意,我們將lambda表達式返回的內(nèi)容賦值給一個名稱(adder_lambda, to_upper_lambda),但是當按照filter()示例中的方式使用lambda時,就不需要這樣做了——不需要把匿名函數(shù)賦給變量。
函數(shù)屬性
Python中每個函數(shù)都是一個完整的對。因此,它有許多屬性。其中一些是特殊的,可以以內(nèi)省的方式在運行時檢查函數(shù)對象。下面的示例,展示了它們的一部分以及如何為示例函數(shù)顯示它們的值:
我們使用內(nèi)置的getattr()函數(shù)來獲取這些屬性的值。getattr(obj, attribute)等價于obj.attribute,當我們需要在運行時動態(tài)地獲取屬性時,就從變量中獲取屬性的名稱(如本例中所示),此時它就會派上用場。
運行這個腳本會得到類似如下輸出:
__doc__ -> 返回a乘以b的結(jié)果. __name__ -> multiplication __qualname__ -> multiplication __module__ -> __main__ __defaults__ -> (1,) __code__ -> <……> __globals__ -> {…略…} __dict__ -> {} __closure__ -> None __annotations__ -> {} __kwdefaults__ -> None |
這里省略了__globals__屬性的值,內(nèi)容太多。這個屬性的含義可以在Python數(shù)據(jù)模型文檔頁面(或自帶幫助文檔中)的可調(diào)用類型部分找到:
??https://docs.python.org/3/reference/datamodel.html#the-standard-typehierarchy??
再次提醒:如果你想查看對象的所有屬性,只需調(diào)用dir(object_name),將得到其所有屬性的列表。
內(nèi)置函數(shù)
Python自帶很多內(nèi)置函數(shù)。它們可以在任何地方使用,你可以通過dir(__builtins__)來查看builtins模塊,或通過訪問官方Python文檔來獲得它們的列表。這里就不一一介紹了。在前面的學習過程中,我們已經(jīng)見過其中的一些,如any、bin、bool、divmod、filter、float、getattr、id、int、len、list、min、print、set、tuple、type和zip等,但還有更多,建議你至少應該閱讀一次。熟悉它們,嘗試它們,為它們每個編寫一小段代碼,并確保您隨時可以使用它們,以便在需要時使用它們。
可在官方文檔中找到這個內(nèi)置函數(shù)列表:https://docs.python.org/3/library/functions.html 。
文檔化代碼
我們非常喜歡不需要文檔的代碼。當我們正確地編程、選擇正確的名稱、并注意細節(jié)時,代碼應該是不言自明的,幾乎不需要文檔。不過,有時注釋非常有用,添加一些文檔化描述也是如此。你可以在Python的PEP 257規(guī)范——文檔字符串約定中找到Python的文檔指南:
??https://www.python.org/dev/peps/pep-0257/,??
但在這里還是會向你展示基本原理。Python的文檔中包含字符串,這些字符串被恰當?shù)胤Q為文檔字符串(docstrings)。任何對象都可以被文檔化來加以描述記錄,可以使用單行或多行文檔字符串。單行程序非常簡單。不是為函數(shù)提供另外的簽名,而應該聲明或描述函數(shù)的目的。請看下面的示例:
使用三重雙引號字符串可以在以后輕松展開或擴展文檔內(nèi)容。
使用以句號結(jié)尾的句子,不要在前后留下空行。
多行注釋的結(jié)構(gòu)與此類似。應該用一行代碼簡單地說明對象的主旨,然后是更詳細的描述。
作為多行文檔化的一個例子,我們在下面的例子中使用Sphinx表示法記錄了一個虛構(gòu)的connect()函數(shù)及文檔化描述:
提示:
Sphinx是用于創(chuàng)建Python文檔的最廣泛使用的工具之一——事實上,官方Python文檔就是用它編寫的。絕對值得花點時間去看看。
內(nèi)置函數(shù)help()用于即時交互使用的,它就使用對象的文檔字符串為對象創(chuàng)建文檔頁面來展示對象的用法。基本用法如下:
首先明確或定義一個對象或函數(shù)(包括已有的對象或函數(shù)),然后使用內(nèi)置help函數(shù),并把對象或函數(shù)做help的參數(shù),該函數(shù)就會返回相應對象的說明文檔了。就這么簡單。
本文小結(jié)
本文主要基于Python語言的一大特色——函數(shù)來拓展的一些相關編程知識,包括遞歸函數(shù)(重點是有限性和邊界性)、lambda函數(shù)(簡潔性和臨時性)以及函數(shù)的屬性以及如何實現(xiàn)函數(shù)的文檔化描述等。