Python中設定變數作為預設值時容易遇到的錯誤

2022-10-04 23:24:32 字數 2027 閱讀 3713

思考一下下面的**片段:

def foo(numbers=):

numbers.append(9)

print numbers

在這裡,我們定義了乙個 list (預設為空),給它加入9並且列印出來。

>>> foo()

[9]>>> foo(numbers=[1,2])

[1, 2, 9]

>>ivvmstng> foo(numbers=[1,2,3])

[1, 2, 3, 9]

看起來還行吧?可是當我們不輸入number 引數來呼叫 foo 函式時,神奇的事情發生了:

>>> foo() # first time, like before

[9]>>> foo() # second程式設計客棧 time

[9, 9]

>>> foo() # third time...

[9, 9, 9]

>>> foo() # what is this black magic?!

[9, 9, 9, 9]

那麼,這是神馬情況?直覺告訴我們無論我們不輸入 number 引數呼叫 foo 函式多少次,這裡的9應該被分配進了乙個空的 list。這是錯www.cppcns.com的!在python裡,函式的預設值實在函式定義的時候例項化的,而不是在呼叫的時候。

那麼我們仍然會問,為什麼在呼叫函式的時候這個預設值卻被賦予了不同的值?因為在你每次給函式指定乙個預設值的時候,python都會儲存這個值。如果在呼叫函式的時候重寫了預設值,那麼這個儲存的值就不會被使用。當你不重寫預設值的時候,那麼python就會讓預設值引用儲存的值(這個例子裡的numbers)。它並不是將儲存的值拷貝來為這個變數賦值。這個概念可能對初學者來說,理解起來會比較吃力,所以可以這樣來理解:有兩個變數,乙個是內部的,乙個是當前執行時的變數。現實就是我們有兩個變數來用相同的值進行互動,所以一旦 numbers 的值發生變化,也會改變python裡面儲存的初始值的記錄。

那麼解決方案如下:

def foo(numbers=none):

if numbers is none:

numbers =

numbers.append(9)

print numbers

通常,當人們聽到這裡,大家會問另乙個關於預設值的問題。思考下面的程式:

def foo(count=0):

count += 1

print count

當我們執行它的時候,其結果完全ivvmstng是我們期望的:

>>> foo()

1>>> foo()

1>>> foo(2)

3>>> foo(3)

4>>> foo()

1ivvmstng

這又是為啥呢?其秘密不在與預設值被賦值的時候,而是這個預設值本身。整型是一種不可變的變數。跟 list 型別不同,在函式執行的過程中,整型變數是不能被改變的。當我們執行 count+=1 這句話時,我們並沒有改變 count 這個變數原有的值。而是讓 count 指向了不同的值。可是,當我們執行 numbers.append(9) 的時候,我們改變了原有的 list 。因而導致了這種結果。

下面是在函式裡使用預設值時會碰到的另一種相同問題:

def print_now(now=time.time()):

print now

跟前面一樣,time.time() 的值是可變的,那麼它只會在函式定義的時候計算,所以無論呼叫多少次,都會返回相同的時間 — 這裡輸出的時間是程式被python解釋執行的時間。

>>> print_now()

1373121487.91

>>> print_now()

1373121487.91

>>> print_now()

1373121487.91

* 這個問題和它的解決方案在 python 2.x 和 3.x 裡都是類似的,在python 3.x 裡面唯一的不同,是裡面的print 表示式應該是函式呼叫的方式(print(numbers))。

本文標題: python中設定變數作為預設值時容易遇到的錯誤

本文位址:

python中不要將列表作為函式引數預設值

def findpath result print result findpath findpath findpath 輸入結果如下 1 1,1 1,1,1 正確的寫法為 def findpath result print result findpath findpath findpath 輸出結果...

Python 使用列表作為函式預設值且為空

python函式中引數預設值為乙個列表且為空時,該列表只初始化一次,而不是像其他程式,每次初始化都為空。3 print my list4 5 printlist 6 printlist 7 printlist 測試結果 2 list list 3 list list list 為了避免這種情況,可以...

在python中設定靜態變數

c語言中,在函式內部可以定義static型別的變數,這個變數是屬於這個函式的全域性物件。在python中也可以實現這樣的機制。def f if not hasattr f,x f.x 0 print f.x f.x 1 f 輸出1 f 輸出2def f class haha cnt 1 def in...