Question 0x01
看看下面的函数会输出什么:
In [157]: def f(x, l=[]):
...: for i in range(x):
...: l.append(i*i)
...: print(l)
...:
In [158]: f(2)
In [159]: f(3,[3,2,1])
In [160]: f(3)
In [161]: f(3)
先仔细想一下,再看下面的答案。
Answer 0x01
In [158]: f(2)
[0, 1]
In [159]: f(3,[3,2,1])
[3, 2, 1, 0, 1, 4]
In [160]: f(3)
[0, 1, 0, 1, 4]
In [161]: f(3)
[0, 1, 0, 1, 4, 0, 1, 4]
第一二个函数执行结果就不说了,大家可能在第三四个结果上感到很奇怪。
大家想到的结果可能是[0, 1, 4]
和[0, 1, 4]
,但实际结果是[0, 1, 0, 1, 4]
和[0, 1, 0, 1, 0, 1, 4]
,原因就是f
函数中的默认参数l
是可变类型。
在第一次调用f
即f(2)
时,l
的默认值[]
已经被改变成了[0, 1]
(相当于默认参数已经被修改了)。
在执行f(3,[3,2,1])
时,l
指向了新的参数[3,2,1]
,但是之前被修改过的默认参数[0,1]
仍然存在(但是[]
不存在了)。
于是,在执行第一个f(3)
时,默认参数l
指向了[0,1]
,并在此基础上添加了0, 1, 4
,因此得到结果为[0, 1, 0, 1, 0, 1, 4]
。第四个执行结果类似。
小结:
不要轻易使用可变对象(列表,集合,字典)作为默认参数使用,除非你知道你在做什么。