预期结果
1 ---1---
2 [0, 1, 4, 9]3 ---2---
4 [0, 1, 4, 9, 16]
实际执行结果
1 ---1---
2 [0, 1, 4, 9]3 ---2---
4 [0, 1, 4, 9, 0, 1, 4, 9, 16]
出坑
当定义函数时,会保存函数中默认参数 list 的值,也就是列表 li=[];
在每次调用的时候如果传递了新的列表,则使用传递的列表,没有传递,使用定义函数时保存的默认参数(li=[]);
上面两次调用中,都没有传递新的列表(使用默认列表 li=[] ),程序会调用定义函数时保存的默认参数((li=[]));
列表在append的时候会在 li=[] 原来的基础上append追加值,所以会产生以上结果.
通过打印列表的ID进行辨识
打印列表 li=[] 的ID:
1 def f(x,li=[]):2 print(id(li)) # 添加打印id3 for i inrange(x):4 li.append(i*i)5 print(li)6
7
8 print('---1---')9 f(4)10 print('---2---')11 f(5)
结果:
1 ---1---
2 140306123906248
3 [0, 1, 4, 9]4 ---2---
5 140306123906248
6 [0, 1, 4, 9, 0, 1, 4, 9, 16]
会发现ID值是相同的; 说明两次执行时使用的都是定义函数时的默认参数 li=[ ]
执行时往里面传新的列表
打印列表 li=[] 的ID 和 传的新列表的ID:
1 def f(x,li=[]):2 print(id(li))3 for i inrange(x):4 li.append(i*i)5 print(li)6
7
8 print('---1---')9 f(4)10 print('---2---')11 f(5,[])12 print('---3---')13 f(6)
结果:
1 ---1---
2 [0, 1, 4, 9]3 ---2---
4 [0, 1, 4, 9, 16]5 ---3---
6 [0, 1, 4, 9, 0, 1, 4, 9, 16, 25]
会发现执行传递空(新)列表的函数时打印的ID不一样,而没有传递的一样;
当传递空列表时,函数体当中会使用传递的空列表,没有传递时,使用函数默认值 li=[ ], 所以会产生以上结果。
优化
如果想要达到预期的结果,只需要在函数体里进行判断即可:
1 def f(x, li=[]):2 if notli:3 #如果li不为空的话,就往下走(清空列表); 为空就不走
4 li =[]5 for i inrange(x):6 li.append(i *i)7 print(li)8
9
10 print('---1---')11 f(4)12 print('---2---')13 f(5)14 print('---3---')15 f(6)
结果:
1 ---1---
2 [0, 1, 4, 9]3 ---2---
4 [0, 1, 4, 9, 16]5 ---3---
6 [0, 1, 4, 9, 16, 25]
----over----
福利赠送: 代码运行自动可视化网站,强烈推荐,不懂就把代码扔进去,它会一步步教给您代码运行的过程。
参考修改:https://www.cnblogs.com/gpd-Amos/p/8998059.html