python循环删除列表元素常见错误与正确方法

常见错误一:使用固定长度循环删除列表元素

l = ['a','b','c']
for i in range(len(l)):
    l.pop(i)

 

报错:
ValueError: list.remove(x): x not in list

原因:
在删除list中的元素后,list的实际长度变小了,但是循环次数没有减少,依然按照原来list的长度进行遍历,所以会造成索引溢出。是的,i的值是一开始就生成好了(0,1,2),后面不会因为列表大小变化了再变化。

图示:

执行第一次pop前,i是0

 


 

执行第一次pop后,删除了索引0原来的a

 

 


 


执行第二次pop前,i是1

 

 


 


执行第二次pop后,删除了索引1原来的c

 

 


 


执行第三次pop前,i是2。继续执行pop,索引2已超出了列表范围(此时列表只有索引0了)。

 

 


 

解决方法:倒序循环遍历

l = ['a','b','c']
for i in range(len(l)-1, -1, -1):
    l.pop(i)

 

 

常见错误二:正序循环遍历删除列表元素

l = ['a','b','c']
for i in l:
    l.remove(i)
print l

 

结果:
['b']

原因:
按顺序循环删除,怎么会有一个没删除呢?原因同上,i的值是一开始就生成好了(0,1,2),后面不会因为列表大小变化了再变化。但是列表因为值的删除,是在变化的。

图示:
执行第一次remove前,i是索引0的a

 

 


 


执行第一次remove后,i是索引0的a被删除

 

 


 

执行第二次remove前,i是索引1的c

 

 


 


执行第二次remove后,i是索引1的c被删除。之后继续执行i是索引2,此时列表已经没有索引2了,循环退出。最后b就被剩下了。

 

 


解决方法一:倒序循环遍历:

l = ['a','b','c']
for i in l[::-1]:
    l.remove(i)
print l

 

解决方法二:深拷贝一个ll,对l遍历,对ll操作

l = ['a','b','c']
ll = l[:]
for i in l:
    ll.remove(i)

此方法的缺点是对于过大的list,拷贝后可能很占内存。那么对于这种情况,可以用倒序遍历的方法来实现。

 

---------------------------------------------------------------------------------

关注微信公众号(测试工程师小站)即可在手机上查阅,并可接收更多测试分享,发送【测试资料】更可获取百G测试教程~

posted @ 2020-05-30 16:12  ☆星空物语☆  阅读(1021)  评论(0编辑  收藏  举报