注意了!ArrayList 增删千万不要乱用…
编程过程中常常需要使用到集合,而arraylist是我们常常使用的,但是最近在一次删除和增加中出现了一些问题,分享记录下。
分下如下俩段代码
list<string> arraylist1 = new arraylist<string>(); arraylist1.add("1"); arraylist1.add("2"); for (string s : arraylist1) { if("1".equals(s)){ arraylist1.remove(s); }} list<string> arraylist2 = new arraylist<string>(); arraylist2.add("2");arraylist2.add("1"); for (string s : arraylist2) { if("1".equals(s)){ arraylist2.remove(s); } }
程序运行结果如下:
arraylist1的remove方法成功执行,
arraylist2的remove方法运行抛出concurrentmodificationexception异常。
我们查看源代码来分析异常原因
因为foreach的本质就是使用迭代器iterator,所有的collecation集合类都会实现iterable接口。
找到arraylist类的iterator()方法
public iterator<e> iterator() { return new itr(); }
迭代器的本质是先调用hasnext()方法判断存不存在下一元素,然后再使用next()方法取下一元素
public boolean hasnext() { return cursor != size; } @suppresswarnings("unchecked") public e next() { checkforcomodification(); int i = cursor; if (i >= size) throw new nosuchelementexception(); object\[\] elementdata = arraylist.this.elementdata; if (i >= elementdata.length) throw new concurrentmodificationexception(); cursor = i + 1; return (e) elementdata\[lastret = i\]; }
上面arraylist1为什么能remove成功呢?其实它只循环了一次,所以成功了。
因为它在remove元素1之后,它的size-1变成1,然后itr内部的cursor变量由0变成1,此时1=1,循环结束,所以成功了。
arraylist2为什么remove失败呢?因为它在循环第二次的时候,也remove成功了,但是第三次判断next的时候cursor的值为2导致不等于现在的size 1,所以执行了next方法,最重要的来了,之前remove的操作导致arraylist的modcount值加1,然后itr类中的expectedmodcount保持不变,所以会抛出异常。
final void checkforcomodification() { if (modcount != expectedmodcount) throw new concurrentmodificationexception(); }
同理可得,由于add操作也会导致modcount自增,所以不允许在foreach中删除, 增加,修改arraylist中的元素。
对此,推荐大家使用迭代器iterator删除元素。
iterator<string> ite = arraylist2.iterator(); while(ite.hasnext()) { if("1".equals(ite.next())) { ite.remove(); } }
如果存在并发操作,还需要对iterator进行加锁操作。
作者:奋斗的小程序员
https://www.toutiao.com/i6754322606561690116/
推荐去我的博客阅读更多:
2.spring mvc、spring boot、spring cloud 系列教程
3.maven、git、eclipse、intellij idea 系列工具教程
生活很美好,明天见~
上一篇: Bootstrap的modal拖动效果