Python坑:不要使用可变对象作为函数默认值、生成器不保留迭代过后的结果、嵌套列表创建、==和is的区

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

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测试教程~

原文地址:https://www.cnblogs.com/songzhenhua/p/13192473.html