`
carvin
  • 浏览: 213429 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

ConcurrentModificationException主要原因及处理方法

    博客分类:
  • java
阅读更多
ConcurrentModificationException主要原因及处理方法
当使用 fail-fast iterator 对 Collection 或 Map 进行迭代操作过程中尝试直接修改 Collection / Map 的内容时,即使是在单线程下运行, java.util.ConcurrentModificationException 异常也将被抛出。

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

  所以 Iterator 在工作的时候是不允许被迭代的对象被改变的。但你可以使用 Iterator 本身的方法 remove() 来删除对象, Iterator.remove() 方法会在删除当前迭代对象的同时维护索引的一致性。

  有意思的是如果你的 Collection / Map 对象实际只有一个元素的时候, ConcurrentModificationException 异常并不会被抛出。这也就是为什么在 javadoc 里面指出:
it would be wrong to write a program that depended on this exception for its correctness: ConcurrentModificationException should be used only to detect bugs.

import java.util.*; 
 
public final class MyTest 
{ 
 private static HashMap p_mapList = new HashMap(2); 
 private MyTest(){} 
 public static void init(){       // If only there are more than one element in Map, 
      // the ConcurrentModificationException will not be
      // thrown.
      p_mapList.put(new String("hello"),new String("world")); 
      p_mapList.put(new String("goto"),new String("hell")); 
 } 
 public static void clear() throws Exception{ 
      Iterator pTmpKeys = null; 
      Long pTmpKeyLong; 
      pTmpKeys = p_mapList.keySet().iterator(); 
      String pCurKey = null; 
      String pCurObj = null; 
      while(pTmpKeys.hasNext()){ 
             pCurKey = (String) pTmpKeys.next(); 
             pCurObj = (String) p_mapList.get(pCurKey); 
 
             p_mapList.put(pCurKey,null); 
             // You can not remove element in Map object directly.
             //p_mapList.remove(pCurKey);
             // But you can remove current element by iterator itself.
             pTmpKeys.remove(); 
 
             System.out.println(pCurKey + " removed."); 
      } 
       System.out.println(p_mapList.size() + 
        " entries left after iterator."); 
         pTmpKeys = null; 
 } 
 public static void main(String[] args) 
 throws Exception{ 
      MyTest.init(); 
     MyTest.clear(); 
  } 
} 

分享到:
评论

相关推荐

    关于方法iterator()找不着的问题~~~~~

    在Java编程中,`iterator()`方法是集合框架中非常关键的一部分,主要用于遍历集合中的元素。当你遇到“方法iterator()找不着的问题”时,这通常意味着你在尝试使用`iterator()`时遇到了错误,可能是由于以下几个原因...

    集合自定义异常

    例如,可以定义一个字段来存储导致异常的具体原因,或者提供一个方法来获取详细的错误消息。 3. 在代码中,当遇到特定的错误条件时,通过`throw new 自定义异常类(错误信息)`来抛出自定义异常。这样,调用者可以...

    java面试笔记整理,包含java,redis,kafka等

    - **异常处理:** Java强制要求捕获或声明抛出异常,而C++则允许程序员决定是否使用异常处理机制。 - **标准库:** Java的类库更加庞大和丰富,提供了更多的工具和框架支持。 - **安全性:** Java内置了更多的安全机制...

    阿里代码规范练习题目及答案.pdf

    以下是一些主要的知识点: 1. **单元测试数据处理**: - 测试数据应加特殊前缀标识,以便区分于生产数据。 - 测试数据应使用独立的测试库,避免污染主数据库。 - 单元测试产生的脏数据应自动回滚,确保测试环境...

    java 中ArrayList迭代的两种实现方法

    使用Iterator的好处是它可以检测并处理并发修改异常(ConcurrentModificationException),这是在遍历集合时尝试修改集合的常见错误。以下是如何使用Iterator遍历ArrayList: ```java ArrayList<String> arrayList ...

    java集合代码

    ArrayList的主要方法包括: 1. `add(E element)`: 向列表末尾添加一个元素。 2. `add(int index, E element)`: 在指定位置插入元素。 3. `remove(int index)`: 删除指定位置的元素。 4. `get(int index)`: 返回指定...

    百度2019年最新面试题库

    - **POST**方法:用于向指定资源提交数据进行处理请求(如提交表单),不能被缓存,不保留在浏览器历史记录中,可能会改变服务器状态。 #### Interface与abstract类的区别 - **接口(Interface)**: 只能包含抽象...

    黑马Java基础口述总结

    - **问题**:可能导致`ConcurrentModificationException`异常。 - **解决方案**:使用迭代器的`remove()`方法安全地删除元素。 #### 48. HashSet如何保证元素唯一性 - **原理**:通过`hashCode()`和`equals()`方法...

    Java开发手册(嵩山版)灵魂15问.pdf

    《Java开发手册(嵩山版)灵魂15问》中提到的禁止使用Apache Beanutils进行属性复制的原因主要是性能和安全性的考虑。Apache BeanUtils提供了便捷的属性拷贝功能,但在大规模或高性能应用中,其效率相对较低。相比其他...

    java面试必会200题.docx

    - **service()**:处理客户端请求,根据请求方法调用相应的方法(如doGet/doPost)。 - **destroy()**:销毁Servlet,只执行一次。 25. **请简述一下Ajax的原理及实现步骤** - Ajax(Asynchronous JavaScript and...

    好几天解决不了的问题源代码

    例如,可能是在遍历`Map`时没有正确地处理`Iterator`,导致并发修改异常(`ConcurrentModificationException`);或者是在查找或插入键值对时遇到了键的唯一性问题;也可能是性能问题,比如在大量数据上使用了不适合...

    JAVA常见面试题300道

    - **方法重写(Overriding)**:发生在继承关系中的两个类之间,子类重新定义父类中的方法,方法名、返回类型及参数列表必须完全相同。 - **方法重载(Overloading)**:发生在同一个类中,允许存在多个同名方法,但...

    Java面试八股文十万字总结.docx

    当多个线程同时访问一个集合时,如果一个线程修改了集合,而另一个线程正在进行迭代,则抛出`ConcurrentModificationException`异常。 **39. 说说Hashtable 与 HashMap 的区别** - **线程安全性**:`Hashtable`...

    Java面试题

    - `Hashtable`直接使用键的`hashCode()`方法来计算哈希值,而`HashMap`会进一步处理键的哈希码,通常会调用`hash(key.hashCode())`来计算哈希值。 - **空键与空值的支持**: - `Hashtable`不允许键或值为空,而`...

    02-Java集合容器面试题(2020最新版)-重点.pdf

    - 当多个线程并发访问集合时,如果其中一个线程修改了集合结构,则其他正在访问该集合的线程会抛出`ConcurrentModificationException`异常,这就是快速失败机制。 #### 十、确保集合不可被修改 - **方法**:可以...

    【Java面试题】List如何一边遍历,一边删除?

    在Java编程中,遍历并删除List集合是一个常见的操作,但在实际编程中,如果不使用正确的方法,可能会导致`java.util.ConcurrentModificationException`异常。本文主要针对这个面试题,详细讲解如何在遍历List的同时...

    2021-2022计算机二级等级考试试题及答案No.9204.docx

    - **原因**:数据不一致的主要原因是数据冗余,即同一数据在数据库中存在多份拷贝。 - **解决方案**:通过实施数据完整性控制措施,比如唯一约束、外键约束等,来避免数据冗余,从而提高数据的一致性。 #### 知识点...

    Java面试题集合部分.docx

    在Java编程语言中,集合框架是处理对象存储和操作的核心工具。面试中,关于Java集合的知识点涵盖了集合的分类、接口、实现类以及其内部的工作原理。以下是对这些知识点的详细解析: 1. Java集合框架的层次结构: -...

Global site tag (gtag.js) - Google Analytics