java.util.ConcurrentModificationException解决方案

今天碰到了java.util.ConcurrentModificationException这个错误,上网找了找其原因,现总结如下:

示例代码:

public void filterAppendWords(Set<Word> wordSet){
for(Iterator it = wordSet.iterator(); it.hasNext(); ){

                        Word word = (Word ) it.next();
String content = word.getContent();
if(!RegexMatch.specialMatch(content))
continue;
wordSet.remove(word);
}
}

运行出错:java.util.ConcurrentModificationException

从API中可以看到List等Collection的实现并没有同步化,如果在多 线程应用程序中出现同时访问,而且出现修改操作的时候都要求外部操作同步化;调用Iterator操作获得的Iterator对象在多线程修改Set的时 候也自动失效,并抛出java.util.ConcurrentModificationException。这种实现机制是fail-fast,对外部 的修改并不能提供任何保证。

网上查找了关于Iterator的原理: Iterator 是工作在一个独立的线程中,并且拥有一个 mutex 锁。 Iterator 被创建之后会建立一个指向原来对象的内存索引表(单链索引表),这个索引表指向的是最初始的对象,当原来的对象数量发生变化时,这个索引表的内容不会同步改变,所以当索引指针往后移动的时候就找不到要迭代的对象,所以按照 fail-fast 原则 Iterator 会马上抛出 java.util.ConcurrentModificationException 异常。

List、Set等是动态的,可变对象数量的数据结构,但是Iterator则是单向不可变,只能顺序读取,不能逆序操作的数据结构,当 Iterator指向的原始数据发生变化时,Iterator自己就迷失了方向。

解决方法:

1、新建一个set用于存放结果集

public static Set<Word> filterAppendWords(Set<Word> wordSet){
Set<Word> resultWords = new HashSet<Word> ();
for(Word word : wordSet){
String content = word.getContent();
if(RegexMatch.specialMatch(content))
continue;
resultWords.add(word);



}
return resultWords;
}

2、使用 Iterator 本身的方法 remove() 来删除对象, Iterator.remove() 方法会在删除当前迭代对象的同时维护索引的一致性。

   

	public static void main(String[] args) {
//		List<Integer> s1 = new ArrayList<Integer> ();
		Set<Integer> s1 = new HashSet<Integer> ();
		for(int index = 0; index < 10; index ++)
			s1.add(index);
		for(Iterator it = s1.iterator(); it.hasNext();  ){
			Integer d = (Integer) it.next();
			if(d % 2 == 0){
				it.remove();
			}
		}
		System.out.println(s1.toString());
	}




### 解决方案 当遇到 `java.util.ConcurrentModificationException` 异常时,通常是因为尝试在一个线程修改集合的同时另一个线程正在迭代该集合。为了防止这种情况发生并解决问题,可以采取以下几种方法: #### 使用 Iterator 的 remove 方法 如果需要在遍历过程中移除元素,则应使用 `Iterator` 提供的安全删除操作而不是直接调用集合的方法。 ```java List<String> list = new ArrayList<>(); // 填充列表... Iterator<String> iterator = list.iterator(); while (iterator.hasNext()) { String item = iterator.next(); if (someCondition(item)) { iterator.remove(); // 安全地移除元素 } } ``` #### 利用并发安全的数据结构 对于多线程环境下的访问需求,可以选择采用线程安全的实现类来替代普通的集合对象,比如 `CopyOnWriteArrayList` 或者 `ConcurrentHashMap` 等。 ```java import java.util.concurrent.CopyOnWriteArrayList; CopyOnWriteArrayList<String> safeList = new CopyOnWriteArrayList<>(); safeList.add("item"); for(String s : safeList){ System.out.println(s); } ``` #### 复制原集合再做改动 另一种方式是在执行任何变更之前先创建原始数据的一个副本,在新的实例上完成必要的更新动作后再替换旧版本的内容。 ```java List<String> original = Arrays.asList("a", "b", "c"); List<String> copy = new ArrayList<>(original); // 创建副本来处理 copy.addAll(Arrays.asList("d","e")); // 对副本进行更改 original = copy; // 更新引用指向新列表 ``` 以上三种策略都可以有效地避免因并发修改而导致的异常情况[^1]。 ### 数据库插入场景中的应用 针对特定于 MyBatis 持久层框架的情况,当批量插入记录到数据库时触发此错误可能是由于事务管理不当造成的。确保每次提交前只构建一次 SQL 语句,并且在整个批处理期间保持相同的 Session 实例非常重要。另外也可以考虑调整批量大小以减少内存占用和提高性能效率[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值