**
一个函数参数的默认值,仅仅在该函数定义的时候,被赋值一次
**
首先来看问题:
>>> def add_end(L=[]):
... L.append('END')
... return L
...
>>>
>>> t=add_end()
>>>
>>> t
['END']
>>> t=add_end()
>>> t
['END', 'END']
>>> t=add_end([1])
>>> t
[1, 'END']
>>> t=add_end()
>>> t
['END', 'END', 'END']
>>> t=add_end([1])
>>> t
[1, 'END']
>>> t=add_end()
>>> t
['END', 'END', 'END', 'END']
Python函数在定义的时候(只在那个时刻赋值,此后调用不再赋值),默认参数L的值就被计算出来了,即[],因为默认参数L也是一个变量,它指向对象[],每次调用该函数,如果改变了L的内容,则下次调用时,默认参数的内容就变了,不再是函数定义时的[]了
Python函数参数对于可变对象,函数内对参数的改变会影响到原始对象;对于不可变对象,函数内对参数的改变不会影响到原始参数。原因在于:
1、可变对象,参数改变的是可变对象,其内容可以被修改。
2、不可变对象,改变的是函数内变量的指向对象。
更明显的例子:可以看出初始化时机
import datetime as dt
def log_time(message, time=dt.datetime.now()):
print("{0}: {1}".format(time.isoformat(), message))
每次我们调用log_time()时都期望它能够正确提供当前时间。悲剧的是并没有成功:默认参数在定义时求值(比如说当你首次导入模块时),调用的结果如下
>>> log_time("message 1")
2015-02-10T21:20:32.998647: message 1
>>> log_time("message 2")
2015-02-10T21:20:32.998647: message 2
>>> log_time("message 3")
2015-02-10T21:20:32.998647: message 3
对于不可变对象,他的内内容发生改变时,变量的对象引用是不会变化的,所以上面的L一直是同一个地址的L,所以会不断的家END 而当改成如下形式:
def add_end(L=None):
if L is None:
L = []
L.append('END')
return L
现在,无论调用多少次,都不会有问题。
None 是不可变对象,每次一赋值为[]时,他指向的引用就会发生改变,输出的结果也是改变后的引用内容,但再一次调用L还是之前的引用(None)
所以,定义默认参数要牢记一点:默认参数必须指向不变对象!