Python中如何避免默認(rèn)參數(shù)的陷阱
在 Python 中,我們經(jīng)常會使用函數(shù)的默認(rèn)參數(shù)來簡化代碼。但你知道嗎?默認(rèn)參數(shù)的行為有時可能會導(dǎo)致一些難以察覺的錯誤,尤其是當(dāng)默認(rèn)參數(shù)是一個可變對象(如列表或字典)時。今天,我們就來探討一下這個問題,并學(xué)習(xí)如何避免它。
問題背景
我們通常會定義一個函數(shù),并為其提供一些默認(rèn)參數(shù)。例如:
def test(a, lst=[1, 2]): """ 給列表添加元素 :param a: 需要添加的元素 :param lst: 需要添加元素的列表 :return: 添加之后的列表 """ if a not in lst: lst.append(a) return lst print(test(4)) # [1, 2, 4] print(test(40)) # [1, 2, 40],實(shí)際輸出:[1, 2, 4, 40] print(test(40, lst=[])) # [40] print(test(80)) # 實(shí)際輸出:[1, 2, 4, 40, 80]
我們在定義函數(shù)的時候,創(chuàng)建的默認(rèn)參數(shù)lst,會在內(nèi)存中創(chuàng)建一塊存儲區(qū)域用來存放這個列表
當(dāng)我們第一次調(diào)用的時候,由于列表是可變類型,我們修改的是實(shí)際的值,但是我們的映射地址沒有變
這個時候該地址對應(yīng)的數(shù)據(jù)就發(fā)生了變化
當(dāng)我們給這個lst傳值得時候,相當(dāng)于我們又新開了一塊內(nèi)存,來儲存這個新的列表,這時候會生成一個新的地址
固當(dāng)我們再次使用test(80)調(diào)用的時候,操作的仍是初始創(chuàng)建的列表
這個函數(shù)的目的是:接收一個參數(shù) a,然后將其添加到列表 lst 中。如果 a 不在列表中,就將它添加進(jìn)去并返回修改后的列表。
我們來看一下這個代碼的執(zhí)行過程。
默認(rèn)參數(shù)的行為
在 Python 中,函數(shù)的默認(rèn)參數(shù)值是 在函數(shù)定義時就計算并初始化的。這意味著,默認(rèn)參數(shù) lst=[1, 2] 只會在函數(shù)首次定義時創(chuàng)建一次,而不是每次調(diào)用函數(shù)時都會重新創(chuàng)建。如果你修改了默認(rèn)參數(shù)中的可變對象(比如列表、字典等),下次調(diào)用函數(shù)時這個對象會被“記住”。
示例分析
print(test(4)) # [1, 2, 4] print(test(40)) # [1, 2, 4, 40] print(test(40, lst=[])) # [40] print(test(80)) # [1, 2, 4, 40, 80]
第一行: 調(diào)用 test(4) 時,lst 使用了默認(rèn)參數(shù) [1, 2],因此 4 被成功地添加到了列表中,返回 [1, 2, 4]。
第二行: 調(diào)用 test(40) 時,由于默認(rèn)參數(shù) lst 被修改過了,變成了 [1, 2, 4],因此 40 被加入到了列表中,返回 [1, 2, 4, 40]。
第三行: 調(diào)用 test(40, lst=[]) 時,你明確傳入了一個新的空列表 [],因此返回的是新的列表 [40],不會受到前面修改的影響。
第四行: 再次調(diào)用 test(80) 時,lst 已經(jīng)被修改為 [1, 2, 4, 40],因此 80 被添加到列表中,最終返回 [1, 2, 4, 40, 80]。
問題的根源
問題的根源在于:默認(rèn)參數(shù)是共享的。當(dāng)你修改了默認(rèn)參數(shù)(如列表、字典等可變對象)時,這些修改會影響到后續(xù)的函數(shù)調(diào)用。
在上面的代碼中,默認(rèn)列表 lst 是可變的,每次調(diào)用函數(shù)時都會操作同一個列表,從而導(dǎo)致了不可預(yù)期的結(jié)果。
如何避免默認(rèn)參數(shù)引發(fā)的問題?
為了避免默認(rèn)參數(shù)引發(fā)的問題,我們可以使用 None 作為默認(rèn)參數(shù),并在函數(shù)內(nèi)部檢查參數(shù)是否為 None,如果是,則創(chuàng)建一個新的列表。修改后的代碼如下:
def test(a, lst=None): if lst is None: lst = [1, 2] # 創(chuàng)建新的列表 if a not in lst: lst.append(a) return lst
這種做法能夠確保每次調(diào)用 test 函數(shù)時,都使用一個全新的列表,而不會出現(xiàn)共享默認(rèn)參數(shù)的問題。
總結(jié)
在 Python 中,默認(rèn)參數(shù)如果是可變對象(如列表、字典等),則它們的修改會影響到后續(xù)的調(diào)用。這種行為可能導(dǎo)致一些難以察覺的錯誤。為了解決這個問題,我們可以將默認(rèn)參數(shù)設(shè)置為 None,然后在函數(shù)內(nèi)部根據(jù)需要創(chuàng)建新的對象。這是避免共享可變對象的經(jīng)典做法。
到此這篇關(guān)于Python中如何避免默認(rèn)參數(shù)的陷阱的文章就介紹到這了,更多相關(guān)Python默認(rèn)參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python之批量創(chuàng)建文件的實(shí)例講解
今天小編就為大家分享一篇Python之批量創(chuàng)建文件的實(shí)例講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-05-05python pygame實(shí)現(xiàn)2048游戲
這篇文章主要為大家詳細(xì)介紹了python pygame實(shí)現(xiàn)2048游戲,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-11-11基于keras輸出中間層結(jié)果的2種實(shí)現(xiàn)方式
今天小編就為大家分享一篇基于keras輸出中間層結(jié)果的2種實(shí)現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-01-01Python實(shí)現(xiàn)學(xué)校管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Python實(shí)現(xiàn)學(xué)校管理系統(tǒng),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-01-01