JAVA List 一边遍历一边删除元素
2024-10-11 06:32:09
JAVA List 一边遍历一边删除元素,报java.util.ConcurrentModificationException异常
在使用set/map时,一个可爱的小bug:java.util.ConcurrentModificationException
【错误场景1】:set容器,边遍历,边add/remove元素
Set<String> set = new HashSet<String>();for (int i = 0; i < 10000; i++) {set.add(Integer.toString(i));}for (String str : set) { //或使用iterator来循环,JDK5.0以上,这样的遍历底层也都是iterator实现。set.add("xxx"); //报错// set.remove(str); //报错}
【错误场景2】:map容器,边遍历,边remove元素
Map<String, String> map = new HashMap<String, String>();for (int i = 0; i < 100; i++) {map.put(Integer.toString(i), Integer.toString(i));}for (String str : map.keySet()) {//或使用iterator来循环map.remove(str); //报错}
【错误场景3】list容器,边遍历,边add/remove元素
List<String> list = new ArrayList<String>();for (int i = 0; i < 100; i++) {list.add(Integer.toString(i));}for (Iterator<String> it = list.iterator(); it.hasNext();) {String val = it.next();if (val.equals("5")) {list.add(val); //报错// list.remove(val); //报错}}
【错误原因】
- 对于remove操作,list.remove(o)的时候,只将modCount++,而expectedCount值未变,那么迭代器在取下一个元素的时候,发现该二值不等,则抛ConcurrentModificationException异常。
- 对于add操作,同remove
【解决办法】
- remove:用iterator提供的原生态remove()
add:同remove就错了,iterator没有提供原生的add()方法。真是的,还要用新的容器暂存,然后再遍历结束后,全部添加到原容器当中。
set/list:这两类常用容器,就用上面说的方法remove(), add()就好了。
map:直接使用ConcurrentHashMap就ok。为什么别的容器,不也实现个concurrent版本直接用。。?库里不搞,自己搞。
【正确使用案例】
for (Iterator<String> it = list.iterator(); it.hasNext();) {String val = it.next();if (val.equals("5")) {it.remove();}}
List<String> newList = new ArrayList<String>();for (Iterator<String> it = list.iterator(); it.hasNext();) {String val = it.next();if (val.equals("5")) {newList.add(val);}}list.addAll(newList);
转载自:http://www.cnblogs.com/alipayhutu/archive/2012/04/23/2465981.html
正确使用案例二:
- public class Test {
- public static void main(String[] args) {
- User user1 = new User();
- user1.setId(1);
- user1.setName("shangsan");
- User user2 = new User();
- user2.setId(2);
- user2.setName("lisi");
- List<User> list = new ArrayList<User>();
- list.add(user1);
- list.add(user2);
- System.out.println("userSet.size() before--------------"+list.size());
- List<User> delList = new ArrayList<User>();
- for (Iterator<User> it = list.iterator(); it.hasNext();) {
- User user = (User) it.next();
- if (user.getId() == 1) {
- delList.add(user);
- }
- }
- list.removeAll(delList);
- System.out.println("userSet.size() after--------------"+list.size());
- }
- }
最新文章
- zookeeper集群
- uva10870 矩阵
- BZOJ4411——[Usaco2016 Feb]Load balancing
- android之简单图形绘制
- JQUERY解析XML IE8的兼容问题
- mac 终端常见指令
- python3.3 基础 新特性
- android raw与assets资源
- I2C分析三
- yii2的变量是如何注入到视图中去的?
- 201521123035《Java程序设计》第十二周学习总结
- C++有符号和无符号数的转换
- 以太坊RPC机制与API实例
- C# 返回JSON格式化统一标准
- LOJ#2245 魔法森林
- python循环语句与其他编程语言不同之处
- 微信小程序 --- e.currentTarget.dataset.id 获取不到值
- Python 爬虫-股票数据的Scrapy爬虫
- alias和alias_method的区别:
- 用Maven构建Mahout项目
热门文章
- cf776D Mahmoud and a Dictionary
- Hive 函数之内置运算符
- 20145202马超 2016-2017-2 《Java程序设计》第四周学习总结
- VI的配置
- vi编辑图
- 【js笔记】数组那些事[0]
- 【app.json】配置说明,不断更新中
- python学习总结----内置函数及数据持久化
- [spring+tomcat]启动时报错:NoSuchMethodError: javax.servlet.http.HttpServletResponse.getStatus()I
- 问题 A: Least Common Multiple