● 不要使用可变对象作为函数默认值
def append(value, arr=[]): arr.append(value) return arr print append(1) print append(1) print append(1)
运行结果:
[1] [1, 1] [1, 1, 1]
原因是模块在加载的时候,arr定义指向了一个空的列表的地址,以后再次调用时,不是我们理解的再给arr分配
一个新的空的列表,而是一直指向的同一个地址的列表,如果列表内容变了其引用也跟着变了。定义默认参数要
牢记一点:默认参数必须指向不变对象(字典,集合,列表等)!
解决方法:
def append(value, arr=None): if arr is None: arr=[] arr.append(value) return arr print append(1) print append(1) print append(1)
运行结果:
[1] [1] [1]
● 生成器不保留迭代过后的结果
gen = (i for i in range(5)) print (1 in gen) print (1 in gen)
运行结果:
True
False
原因是生成器中的值一旦使用过后,就不再保留了。这个是生成器的原理导致的,生成器不像列表,一开始就把
值都初始化好了,而是每次用到的时候生成一个。
解决方法:
gen = (i for i in range(5)) arr = list(gen) # 转成list或tuple print (1 in arr) print (1 in arr)
运行结果:
True
True
● 嵌套列表创建
li = [[]]*3 print li li[0].append(1) print li
运行结果:
[[], [], []]
[[1], [1], [1]]
上面是生成了3个list的嵌套list,只给list中的第1个list赋值,但运行结果发现赋值对所有的嵌套list都生效
了。这是因为[[]]*3并不是创建了三个不同list,而是创建了三个指向同一个list的对象,也就是list
[0],list[1],list[2]都是指向同一地址的list。
我们看下内存运行原理(工具详见https://mp.weixin.qq.com/s/IoIOiNDyeaIpWxgxpo1a3g)
解决方法:
li = [[] for _ in range(3)] li[0].append(1) print li
运行结果:
[[1], [], []]
● == 和 is 的区别
print (255 + 1) is (255 + 1) print (256 + 1) is (256 + 1)
运行结果:
True
False
默认情况下,[-5,256]会在解释器第一次启动时创建并缓存,所以才会有上面的奇怪的行为。这是个很常见但很
容易被忽略的一个坑。解决方案是始终使用equality(==)运算符而不是 identity(is)运算符比较值。
is用来判断地址和值是否都相同。
==比较两个对象内容是否相等,内存地址可以不一样
解决方法:
print (255 + 1) == (255 + 1) print (256 + 1) == (256 + 1)
运行结果:
True
True
--------------------------------------------------------------------------------
关注微信公众号(测试工程师小站)即可在手机上查阅,并可接收更多测试分享,发送【测试资料】更可获取百G测试教程~