Python 新手常犯錯誤(第一部分)

用一個可變的值做爲默認值

這是一個絕對值得放在第一個來講的問題。不單單是由於產生這種BUG的緣由很微妙,並且這種問題也很難檢查出來。思考一下下面的代碼片斷:app

def foo(numbers=[]):
   numbers.append(9)
   print numbers

在這裏,咱們定義了一個 list (默認爲空),給它加入9而且打印出來。函數

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

看起來還行吧?但是當咱們不輸入number 參數來調用 foo 函數時,神奇的事情發生了:code

>>> 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。這是錯的!在Python裏,函數的默認值實在函數定義的時候實例化的,而不是在調用的時候。變量

那麼咱們仍然會問,爲何在調用函數的時候這個默認值卻被賦予了不一樣的值?由於在你每次給函數指定一個默認值的時候,Python都會存儲這個值。若是在調用函數的時候重寫了默認值,那麼這個存儲的值就不會被使用。當你不重寫默認值的時候,那麼Python就會讓默認值引用存儲的值(這個例子裏的numbers)。它並非將存儲的值拷貝來爲這個變量賦值。這個概念可能對初學者來講,理解起來會比較吃力,因此能夠這樣來理解:有兩個變量,一個是內部的,一個是當前運行時的變量。現實就是咱們有兩個變量來用相同的值進行交互,因此一旦 numbers 的值發生變化,也會改變Python裏面保存的初始值的記錄。引用

那麼解決方案以下:程序

def foo(numbers=None):
   if numbers is None:
       numbers = []
   numbers.append(9)
   print numbers

一般,當人們聽到這裏,你們會問另外一個關於默認值的問題。思考下面的程序:im

def foo(count=0):
   count += 1
   print count

當咱們運行它的時候,其結果徹底是咱們指望的:英文

>>> foo()
1
>>> foo()
1
>>> foo(2)
3
>>> foo(3)
4
>>> foo()
1

這又是爲啥呢?其祕密不在與默認值被賦值的時候,而是這個默認值自己。整型是一種不可變的變量。跟 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))。

你們應該注意到我在解決方案裏用了 if  numbers is None 而不是 if not numbers 。這是另外一種常見的錯誤,我準備在接下來的文章裏面介紹。

英文出處:Amir Rachum

譯文出處:伯樂在線 - 賤聖OMG

相關文章
相關標籤/搜索