摘要:源自我的博客里面一個常見的陷阱就是函數(shù)的默認(rèn)參數(shù)問題。默認(rèn)參數(shù)的一個應(yīng)用先看下面的一個經(jīng)典的例子簡略版本結(jié)果是而不是,原因就是閉包的延遲綁定。結(jié)果自然就是這就是默認(rèn)參數(shù)的一個應(yīng)用。
源自: 我的博客?
python 里面一個常見的陷阱就是函數(shù)的默認(rèn)參數(shù)問題。如下:
def func(mylist = []): mylist.append(1) return mylist
以下的執(zhí)行結(jié)果如下:
print func() print func() print func() print func(["a"]) print func()
結(jié)果如下:
[1] [1, 1] [1, 1, 1] ["a", 1] [1, 1, 1, 1]
如此結(jié)果, 前面三個可以看出 如果沒有指定參數(shù)的話, 每次調(diào)用函數(shù)時候, 調(diào)用的mylist 是同一個對象。這是因?yàn)楹瘮?shù)的默認(rèn)參數(shù),是在代碼編譯成PyCodeObject的時候, 就已經(jīng)創(chuàng)建了對象指針,并且存在該函數(shù)的func_default內(nèi)。 以后在代碼運(yùn)行,調(diào)用函數(shù)的時候,如果沒有指定參數(shù)的話, 每次調(diào)用的話, 該參數(shù)變量都是代碼編譯階段的變量指針?biāo)付ǖ膶ο蟆?/p>
print func.func_default
此時結(jié)果就是:
([1, 1, 1, 1], )
默認(rèn)參數(shù)分為兩種情況:
默認(rèn)參數(shù)值是不可變對象
此時函數(shù)的 func_default 一直指向該不變對象, 如果函數(shù)內(nèi)部修改了該變量, 那么該默認(rèn)參數(shù)會指向一個新的不可變對象.
不過func_default 不變。 而每次調(diào)用函數(shù)都是讀取func_default, 因此每次執(zhí)行都一樣。
In [30]: def func2(var = 1): ....: var += 1 ....: return var ....: In [31]: func2() Out[31]: 2 In [32]: func2() Out[32]: 2 In [34]: func2.func_defaults Out[34]: (1,)
默認(rèn)參數(shù)是可變對象,比如 list, dict, class等
這種情況下,如果在函數(shù)內(nèi)修改了指針?biāo)傅膶ο螅?strong>并未創(chuàng)建新的對象), 那么 func_default 就會改變。這正是開始的mylist發(fā)生變化的原因。看下面的例子,:
In [35]: def func(mylist = []): ....: mylist = [] #這里 創(chuàng)建了新的對象, mylist.append(1) return mylist In [44]: func() Out[44]: [1] In [45]: func.func_defaults Out[45]: ([],) 由于創(chuàng)建了對象, mylist 只是作為一個 新建對象的別名存在, 后面在修改已經(jīng)與 func_default 無關(guān)了。
默認(rèn)參數(shù)的一個應(yīng)用
先看下面的一個經(jīng)典的例子:
def outer(): res = [] for i in range(4): def inner(j): return j * i res.append(inner) return res print [m(2) for m in outer()] #簡略版本: def multipliers(): return [lambda x : i * x for i in range(4)] print [m(2) for m in multipliers()]
結(jié)果是 [6, 6, 6, 6] , 而不是 [0, 2, 4, 6], 原因就是閉包的延遲綁定。另外函數(shù)綁定的是變量而不是綁定數(shù)值。當(dāng)循環(huán)結(jié)束了,i的值已經(jīng)是3, 此時結(jié)果都是6. 一個解決方法便是,使用默認(rèn)參數(shù)綁定數(shù)值。如下改動:
def outer(): res = [] for i in range(4): def inner(j, i = i): return j * i res.append(inner) return res print [m(2) for m in outer()] #簡略版本: def multipliers(): return [lambda x, i = i : i * x for i in range(4)] print [m(2) for m in multipliers()]
這樣的話, 利用默認(rèn)參數(shù)在代碼編譯的時候,便把參數(shù)寫到函數(shù)的func_default中, 就可以綁定0,1,2,3了。結(jié)果自然就是
[0, 2, 4, 6]
這就是默認(rèn)參數(shù)的一個應(yīng)用。
上述還有一個生成器修改的方式
def multipliers(): return (lambda x : i * x for i in range(4)) #修改成生成器 print [m(2) for m in multipliers()]
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/37787.html
摘要:之機(jī)器學(xué)習(xí)第一彈。機(jī)器學(xué)習(xí)是發(fā)展中應(yīng)用廣泛的一個領(lǐng)域。庫集成了一些常用的機(jī)器學(xué)習(xí)方法在進(jìn)行機(jī)器學(xué)習(xí)任務(wù)時并不需要實(shí)現(xiàn)算法只需要簡單的調(diào)用庫中提供的模塊就能完成大多數(shù)的機(jī)器學(xué)習(xí)任務(wù)。 ...
摘要:前言本篇主要介紹基本數(shù)據(jù)類型,以文本進(jìn)度條為例,介紹庫的使用。 前言 本篇主要介紹基本數(shù)據(jù)類型,以文本進(jìn)度條為例,介紹time庫的使用。 并在最后對蟒蛇繪制的代碼進(jìn)...
摘要:本人年開發(fā)經(jīng)驗(yàn),現(xiàn)就職于電信,因工作需要學(xué)習(xí),記錄自己的學(xué)習(xí)記錄。 本人java10年開發(fā)經(jīng)驗(yàn),現(xiàn)就職于電信,因工作需要學(xué)習(xí)python,記錄自己的學(xué)習(xí)記錄。后面也...
摘要:為我們提供了許多內(nèi)置函數(shù),例如并提供了創(chuàng)建用戶定義函數(shù)的能力。會將該變量視為函數(shù)級作用域中的局部變量。回到目錄中函數(shù)的用途是什么是中的內(nèi)置函數(shù)之一。請注意,這種類型的參數(shù)語法不允許將命名參數(shù)傳遞給函數(shù)。函數(shù)接受一個稱為的可選參數(shù)。 ...
摘要:一般來說,這一例行程序用于處理請求的每一部分,例如把路徑作為一系列字典鍵值進(jìn)行處理。,必須是按照中所規(guī)定地鍵值元組列表。行為時回車換行。這個包裝器也可能用模塊指明那些有問題的,但不完全違反的行為。 wsgirf-WSGI功能及參考實(shí)現(xiàn) 源碼:Lib/wsgiref Web服務(wù)器網(wǎng)關(guān)接口(Web Server Gateway Interface, WSGI),是用Python寫的一個服務(wù)...
閱讀 2112·2023-04-26 00:41
閱讀 1142·2021-09-24 10:34
閱讀 3573·2021-09-23 11:21
閱讀 4031·2021-09-22 15:06
閱讀 1557·2019-08-30 15:55
閱讀 898·2019-08-30 15:54
閱讀 1829·2019-08-30 15:48
閱讀 550·2019-08-29 13:58