`
xinklabi
  • 浏览: 1584915 次
  • 性别: Icon_minigender_1
  • 来自: 吉林
文章分类
社区版块
存档分类
最新评论

迭代的同时删除元素导致java.util.ConcurrentModificationException异常问题及如何解决

    博客分类:
  • Java
 
阅读更多

List迭代并删除没有问题,Map迭代并使用map的remove会抛出异常,但如果使用iterator的remove不会出问题。

List<Integer> list = new ArrayList<Integer>();

list.add(1);

list.add(2);

list.add(3);

 

for(int i=0;i<list.size();i++){

if(list.get(i)%2 == 0)

list.remove(i);

}

System.out.println("list 通过下标遍历并删除");

for(int i=0;i<list.size();i++){

System.out.println(list.get(i));

}

 

 

List<Integer> list1 = new ArrayList<Integer>();

list1.add(1);

list1.add(2);

list1.add(3);

list1.add(4);

list1.add(5);

Iterator<Integer> it = list1.iterator();

System.out.println("list 通过迭代器遍历并删除");

while(it.hasNext()){

if(it.next()%2 == 0)

it.remove();

}

for(int j=0;j<list1.size();j++){

System.out.println(list1.get(j));

}

 

Map<String, String> map = new HashMap<String, String>();

map.put("a", "a");

map.put("b", "b");

map.put("c", "c");

Iterator<String> it1  = map.keySet().iterator();

String key = null;

System.out.println("map 通过迭代器遍历并删除");

for(;it1.hasNext();){

key = it1.next();

if(key.equals("b"))

map.remove(key);

}

 

list 通过下标遍历并删除

1

3

list 通过迭代器遍历并删除

1

3

5

map 通过迭代器遍历并删除

Exception in thread "main" java.util.ConcurrentModificationException

at java.util.HashMap$HashIterator.nextEntry(Unknown Source)

at java.util.HashMap$KeyIterator.next(Unknown Source)

at com.test.TestIterator.main(TestIterator.java:54)

 

 下面这段map迭代并删除元素只要使用iterator的删除就不会抛异常。

  1. import java.util.HashMap;  
  2. import java.util.Iterator;  
  3. import java.util.Map;  
  4.   
  5. public class HashMapTest {  
  6.    private static Map<Integer, String> map=new HashMap<Integer,String>();  
  7.       
  8.    /**  1.HashMap 类映射不保证顺序;某些映射可明确保证其顺序: TreeMap 类 
  9.     *   2.在遍历Map过程中,不能用map.put(key,newVal),map.remove(key)来修改和删除元素, 
  10.     *   会引发 并发修改异常,可以通过迭代器的remove(): 
  11.     *   从迭代器指向的 collection 中移除当前迭代元素 
  12.     *   来达到删除访问中的元素的目的。   
  13.     *   */   
  14.    public static void main(String[] args) {  
  15.         map.put(1,"one");  
  16.         map.put(2,"two");  
  17.         map.put(3,"three");  
  18.         map.put(4,"four");  
  19.         map.put(5,"five");  
  20.         map.put(6,"six");  
  21.         map.put(7,"seven");  
  22.         map.put(8,"eight");  
  23.         map.put(5,"five");  
  24.         map.put(9,"nine");  
  25.         map.put(10,"ten");  
  26.         Iterator<Map.Entry<Integer, String>> it = map.entrySet().iterator();  
  27.         while(it.hasNext()){  
  28.             Map.Entry<Integer, String> entry=it.next();  
  29.             int key=entry.getKey();  
  30.             if(key%2==1){  
  31.                 System.out.println("delete this: "+key+" = "+key);  
  32.                 //map.put(key, "奇数");   //ConcurrentModificationException  
  33.                 //map.remove(key);      //ConcurrentModificationException  
  34.                 it.remove();        //OK   
  35.             }  
  36.         }  
  37.         //遍历当前的map;这种新的for循环无法修改map内容,因为不通过迭代器。  
  38.         System.out.println("-------\n\t最终的map的元素遍历:");  
  39.         for(Map.Entry<Integer, String> entry:map.entrySet()){  
  40.             int k=entry.getKey();  
  41.             String v=entry.getValue();  
  42.             System.out.println(k+" = "+v);  
  43.         }  
  44.     }  
  45. }  
分享到:
评论

相关推荐

    java.util.ConcurrentModificationException 解决方法

    java.util.ConcurrentModificationException 解决方法 在使用iterator.hasNext()操作迭代器的时候,如果此时迭代的对象发生改变,比如插入了新数据,或者有数据被删除。 则使用会报以下异常: Java.util....

    出现java.util.ConcurrentModificationException 问题及解决办法

    在Java编程中,`java.util.ConcurrentModificationException` 是一个常见的运行时异常,通常发生在尝试并发修改集合时。这个异常的产生是由于集合类(如HashMap)的非线程安全特性,当你在一个线程中使用迭代器遍历...

    java.util.concurrent系列文章(1)

    ### Java.util.concurrent 系列文章知识点总结 #### 一、引言 随着多核处理器的普及,多线程编程已成为现代软件开发中的一个重要组成部分。Java 5 引入了 `java.util.concurrent` 包,该包提供了丰富的 API 来简化...

    java 集合并发操作出现的异常ConcurrentModificationException

    在Java编程中,`ConcurrentModificationException`是一个常见的运行时异常,主要出现在多线程环境下对集合类(如List、Set、Map等)进行并发修改时。然而,这个异常不仅限于多线程环境,即使在单线程中,如果在遍历...

    Java语言的Util类详细介绍

    但是,因为Vector是同步的,当一个Iterator被创建而且正在被使用,另一个线程改变了Vector的状态(例如,添加或删除元素),Iterator将抛出ConcurrentModificationException异常。 Set接口是Collection接口的子接口...

    遍历并批量删除容器中元素出现ConcurrentModificationException原因及处置

    但是,如果你在循环内部直接修改集合(比如删除元素),而不是通过迭代器的`remove()`方法,那么迭代器将无法跟踪这些变化,从而导致异常。 - 在多线程环境下,如果多个线程同时对同一个集合进行迭代和修改,由于...

    java Iterator迭代器的使用

    它提供了一种安全的方式来访问集合中的元素,同时允许在遍历过程中删除元素。迭代器的主要方法包括`hasNext()`(检查集合中是否存在下一个元素)、`next()`(返回当前元素并移动到下一个)和`remove()`(删除当前...

    java 迭代及迭代器的小例子

    例如,在java.util.concurrent包中的ConcurrentLinkedQueue等线程安全的集合类,它们的迭代器是弱一致性(weakly consistent)的,这意味着迭代器不会抛出ConcurrentModificationException,但可能不反映对集合的...

    迭代大师的修炼之道:Java中Iterator与增强for循环的深度解析

    - 如果试图在迭代过程中添加元素,会导致`UnsupportedOperationException`异常。这是因为`Iterator`的设计初衷是为了避免在遍历过程中修改集合,从而提高程序的稳定性。 #### 三、Enhanced for loop:现代的迭代...

    JAVA.BUG模式详解

    同时,避免在迭代时修改集合,这可能导致ConcurrentModificationException。 八、I/O操作的优化 使用NIO(New IO)替代传统的IO,可以提高读写速度和系统吞吐量。同时,及时关闭文件流,防止资源浪费。 九、过度...

    多线程中使用Java集合类.doc

    例如,上述代码中,线程在删除ArrayList中的元素时,另一个线程尝试遍历并打印ArrayList,这将导致异常。 快速失败机制是为了防止线程间的不确定性行为,它通过迭代器内部的modCount变量来检测集合是否在迭代过程中...

    使用Iterator接口遍历集合元素

    否则将会引发 java.util.ConcurrentModificationException 异常。 3. Iterator 迭代器采用的是快速失败(fail-fast)机制,一旦在迭代过程中检测到该集合已经被修改(通常是程序中其它线程修改),程序立即引发 ...

    iterator-demo 迭代器设计模式demo

    - 在使用迭代器时,若在遍历过程中修改了聚合对象,可能会影响迭代器的行为,可能导致`ConcurrentModificationException`等异常。 在“iterator-demo”项目中,你可能会看到如何创建一个自定义的聚合类和迭代器类...

    java常见的异常

    11. **NullPointerException**和**NullPointerException**:这两种异常在多线程编程中尤为常见,由于并发访问共享资源可能导致数据不一致,因此应使用`synchronized`关键字或`java.util.concurrent`包中的工具来管理...

    Java源码分析:深入探讨Iterator模式

    为了遍历这些容器中的元素,Java引入了迭代器模式(Iterator Pattern),这是一种常用的设计模式,它提供了一种访问聚合对象的方法,而无需暴露其内部表示。本文将详细解析Iterator模式在Java中的实现原理,并通过...

    java基础 集合-22-迭代器设计模式

    为了解决这个问题,Java提供了`ConcurrentSkipListMap`和`CopyOnWriteArrayList`等线程安全的集合,它们的迭代器允许并发修改。 7. **增强型for循环(foreach)** Java 5引入了增强型for循环,也称为foreach循环,...

    内置迭代器的linked list例题

    通过迭代器,我们可以方便地遍历和修改LinkedList中的元素,同时避免了数组或ArrayList在插入和删除操作时的性能问题。在实际开发中,根据具体需求选择合适的数据结构和遍历方式是提高代码效率的关键。

    Java多线程安全集合

    - `CopyOnWriteArrayList`和`CopyOnWriteArraySet`:这些列表和集在线程安全的迭代器上有优势,因为它们在修改时复制底层数组,从而避免了迭代过程中的并发修改异常(`ConcurrentModificationException`)。...

Global site tag (gtag.js) - Google Analytics