Python的16个“坑”

1. 不要使用可变对象作为函数默认值

代码如下:

12345678910111213141516171819202122232425262728 In [1]: def append_to_list(value, def_list=[]):   ...:         def_list.append(value)   ...:         return def_list   ...: In [2]: my_list = append_to_list(1) In [3]: my_listOut[3]: [1] In [4]: my_other_list = append_to_list(2) In [5]: my_other_listOut[5]: [1, 2] # 看到了吧,其实我们本来只想生成[2] 但是却把第一次运行的效果页带了进来 In [6]: import time In [7]: def report_arg(my_default=time.time()):   ...:         print(my_default)   ...: In [8]: report_arg() # 第一次执行1399562371.32 In [9]: time.sleep(2) # 隔了2秒 In [10]: report_arg()1399562371.32 # 时间竟然没有变



这2个例子说明了什么? 字典,集合,列表等等对象是不适合作为函数默认值的. 因为这个默认值实在函数建立的时候就生成了, 每次调用都是用了这个对象的”缓存”. 我在上段时间的分享python高级编程也说到了这个问题,这个是实际开发遇到的问题,好好检查你学过的代码, 也许只是问题没有暴露。

可以这样改,代码如下:

12345 def append_to_list(element, to=None):    if to is None:        to = []    to.append(element)    return to

2. 生成器不保留迭代过后的结果

代码如下:

1234567891011121314151617181920212223 In [12]: gen = (i for i in range(5)) In [13]: 2 in genOut[13]: True In [14]: 3 in genOut[14]: True In [15]: 1 in genOut[15]: False # 1为什么不在gen里面了? 因为调用1->2,这个时候1已经不在迭代器里面了,被按需生成过了 In [20]: gen = (i for i in range(5)) In [21]: a_list = list(gen) # 可以转化成列表,当然a_tuple = tuple(gen) 也可以 In [22]: 2 in a_listOut[22]: True In [23]: 3 in a_listOut[23]: True In [24]: 1 in a_list # 就算循环过,值还在Out[24]: True

3. lambda在闭包中会保存局部变量

代码如下: