电竞比分网-中国电竞赛事及体育赛事平台

分享

Python中的默認(rèn)參數(shù)值

 機(jī)械設(shè)計與模具 2016-10-05

Python對默認(rèn)參數(shù)值的處理方法是少有的幾個易使大多數(shù)新手Python程序員犯錯的地方之一。(通常只犯一次)

導(dǎo)致困惑的地方是當(dāng)你使用“可變”對象作為(參數(shù)的)默認(rèn)值時的(程序)行為。(可變)也就是說值可以原地修改,像列表或字典。

看下面的例子:


就像你看到的那樣,列表變得越來越長。如果你查看列表的ID,你會發(fā)現(xiàn)函數(shù)實際上總是返回同一個對象。


原因很簡單:函數(shù)在每次調(diào)用時總是使用同一個對象。我們做的修改行為有“粘性”。

為什么會這樣?

屬于函數(shù)定義的默認(rèn)參數(shù)值當(dāng)且僅當(dāng)“def”語句執(zhí)行時求值。請看:

http://docs./ref/function.html (死鏈接)

語言參考中的相關(guān)章節(jié)。

要注意Python中的“def”語句是可執(zhí)行的,而且默認(rèn)參數(shù)是在“def”語句的環(huán)境下求值的。如果“def”語句執(zhí)行了多次,那么它每次將產(chǎn)生新的函數(shù)對象(對象會帶著全新的默認(rèn)值)。下面我們會看到這樣的例子的。

那要怎么做?

迂回方法是,就像其他人已經(jīng)提到了的,使用占位符值來替代默認(rèn)值。None是一個常用的值:


如果你需要處理任意對象(包括None),你可以使用哨兵對象:


在很早以前,即在“object”對象引入之前,有時你會看到像下面這樣的代碼:


這個代碼用于創(chuàng)建一個具有唯一ID的對象;這對中括號([])在每次執(zhí)行時產(chǎn)生新的列表。

對可變默認(rèn)參數(shù)的合法(合理)使用

最后,要提到很多高級的Python代碼經(jīng)常使用這個機(jī)制的好處。例如,假設(shè)你要在一個循環(huán)里創(chuàng)建了一堆UI按鈕,而你可能會使用像下面這樣的代碼:


這樣你會發(fā)現(xiàn)所有的回調(diào)函數(shù)都打印出相同的值(在這個情況下,很可能是9)。原因是Python的嵌套作用域是綁定到變量的,而不是綁定到對象值的。所以所有的回調(diào)函數(shù)實例將會看到當(dāng)前(也是最后)的變量“i”的值。為了修正這個問題,使用下面的代碼:


那個“i=i”的步伐將綁定參數(shù)“i”(一個局部變量)到當(dāng)前的外部變量“i”的值。

兩個其他的例子使用的是局部緩存:


(這對某些遞歸算法很好)

另一個例子,對應(yīng)高度優(yōu)化的代碼,對全局名字的局部重新綁定:


詳細(xì)來說,這是怎么工作的?

當(dāng)Python執(zhí)行“def”語句時,它使用一些已有的東西(包括編譯了的函數(shù)體的代碼和當(dāng)前的名字空間),然后創(chuàng)建出一個新的函數(shù)對象。當(dāng)它做這個的時候,默認(rèn)值也會被求值。

這些各種各樣的組件也能作為函數(shù)對象的屬性而訪問。使用我們先前定義過的函數(shù):


由于你可以訪問它們,因而你也可以修改它們:


當(dāng)然,這可不是我推薦的正常使用方法。

另一個重置默認(rèn)值的方法就是簡單的重新執(zhí)行一下那個相同的“def”語句。Python會重新創(chuàng)建一個新的到這個代碼對象綁定,重新計算默認(rèn)值,然后將這個函數(shù)對象賦值給同以前一樣的那個變量。但是要著重指出,請在你確切知道你在做什么的時候才能這么做。

最后,如果你剛好有函數(shù)各個部分,而不是函數(shù)本身,你可以使用new模塊中的function類來創(chuàng)建你自己的函數(shù)對象。

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多