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

相关文章
相关标签/搜索