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

Posted 星空物语

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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测试教程~

以上是关于Python坑:不要使用可变对象作为函数默认值生成器不保留迭代过后的结果嵌套列表创建==和is的区的主要内容,如果未能解决你的问题,请参考以下文章

python函数默认参数为可变对象的理解

Python之在函数中使用列表作为默认参数

Python开发easy忽略的问题

python的形参为可变数据类型的坑

python 不要使用可变类型作为默认参数

不要使用可变类型作为参数的默认值