实践才是最好的老师,对于程序员来说可能更为重要。
首先列出在编码中遇到的问题,给出一个列表nums=[1,2,3,3,3,4],要求将这个列表中值为3的元素删除,计算剩下列表的长度。
题目并不难,一个思想就是遍历list元素,一旦该元素属于这个列表,从该链表中删除,因为链表是可覆盖对象,遍历完成后计算链表的长度即可。
但是我在编码中发现了一个问题。
nums=[1,2,3,3,3,4]
for i in nums:
if i ==3:
nums.remove(i)
print(len(nums))
看上去程序没有什么问题,但是结果并不是预期中的3,而是4。想过采用下面这种方式,就会得到正确答案。
nums=[1,2,3,3,3,4]
for i in nums[:]:
if i ==3:
nums.remove(i)
print(len(nums))
nums和nums[:]之间的区别,其实就是引用和副本之间的区别,针对可覆盖变量,如果两个变量引用相同的内存,指向同一个对象,修改一个变量的值就会连带的对另一个变量造成影响。这为某些程序的实现提供了方便,同时也容易产生不易发现的BUG。通过创建对应的副本,可以避免这样的影响。
在第一个代码块中,由于nums是可覆盖变量,当遇到nums中的第一个值为3的元素是,这时候i遍历到了nums中的第3个位置,由于符合删除条件,nums执行移除remove操作,这时候nums变成了[1,2,3,3,4],接下来i应该查找nums中的第四个位置,这时候就会忽略在nums里的第一个3,直接跳到了第二个3,完成了remove操作,造成最终结果nums变成[1,2,3,4],所以长度为4。使用nums的副本很好的避免了这个问题。
下面这个程序可以更好的说明问题。
nums=[1,2,[3,3,3],4]
for i in nums:
if type(i) is list:
print(i[:])
for j in i[:]:
i[:].remove(j)
print(i[:])
print(nums)
输出结果:
也不知道我接受的清不清楚,有问题大家一起交流哦。