Hashtable 删除元素, 抛出异常 java.util.ConcurrentModificationException

今天在对一个Hashtable对象进行 搜索 -> 删除 操作时遇到的一个问题,开始的使用我使用的是Hashtable的Iterator,然后直接执行:

Hashtable.remove(key); 抛出异常  java.util.ConcurrentModificationException

查了一下java api doc,相关介绍如下:

由迭代器返回的 Iterator 和由所有 Hashtable 的“collection 视图方法”返回的 Collection 的 listIterator 方法都是快速失败的:在创建 Iterator 之后,如果从结构上对 Hashtable 进行修改,除非通过 Iterator 自身的移除或添加方法,否则在任何时间以任何方式对其进行修改,Iterator 都将抛出 ConcurrentModificationException。因此,面对并发的修改,Iterator 很快就会完全失败,而不冒在将来某个不确定的时间发生任意不确定行为的风险。由 Hashtable 的键和值方法返回的 Enumeration不是快速失败的。

Hashtable<Integer,Integer> t = new Hashtable<Integer,Integer>();
     
     for(int i=0;i<10;i++)
     {
      t.put(i,i*10);
     }   
     
     System.out.println("t.size():" + t.size());
     System.out.println(t.get(6));
     
     System.out.println();       
         
        Iterator<Integer> i = t.keySet().iterator();
     
     while(i.hasNext())
     {
      int key = i.next();
      
      if(key == 3)      
      {
       //t.remove(key);  //这样删除,抛出异常 java.util.ConcurrentModificationException       
       i.remove(); //只有这样,才可以成功的删除一个元素
      }
     }
     
     System.out.println("t.size():" + t.size());     
     System.out.println(t.get(6));

使用上面这种t.keySet().iterator();方式,如果在多线程情况下,也是会抛出 java.util.ConcurrentModificationException

在多线程并发的情况下,在增加、修改、删除、遍历时全加上 synchronized :

public synchronized void delKey(int key)
 {  
  Iterator<Integer> i = t.keySet().iterator();
  while(i.hasNext())
     {
      int abc = i.next();
      
      //t.remove(key); //抛出异常 java.util.ConcurrentModificationException
   
      if(abc == key)
       i.remove(); //只有这样,才可以成功的删除一个元素
     }     
 }

当在多线路中调用 delKey()时,必须在定义delKey()方法前使用:synchronized,以保证多线程调用安全。

---------------------------------------------------------------------------------------------------------------------------------------------------------------------

其实还有一种更早的(更原始的)的安全方法,使用Enumeration,这种方式不管是在单线程还是在多线程中执行都是安全的(注意下面的特别说明):

Hashtable<Integer,Integer> t = new Hashtable<Integer,Integer>();
     
     for(int i=0;i<10;i++)
     {
      t.put(i,i*10);
     }    
     
     System.out.println("t.size():" + t.size());
     System.out.println(t.get(6));
     
     System.out.println();    
     
     Enumeration<Integer> e = t.keys();
     while(e.hasMoreElements())
     {
      int key = e.nextElement();
      
      if(key == 3)
      t.remove(key);
     } 
     
     System.out.println("t.size():" + t.size());     
     System.out.println(t.get(6));

特别注意:

虽然使用Enumeration来遍历元素是线程安全的,高并发的情况下进行增加、修改、遍历,也不会抛什么异常,但如果在遍历Enumeration的同时删除里面的一个元素虽不会抛出什么异常,但结果可能不是你想像的:list = new Vector<String>();
  
  list.add("1");
  list.add("2");
  list.add("3");
  list.add("4");
  list.add("5");
  list.add("6");
  
  Enumeration<String> i = list.elements();
  while(i.hasMoreElements())
  {
   String str = i.nextElement();
   
   System.out.println(str);
   
   list.removeElement(str); //在这里删除
  }

//输出结果:

1

3

5

我看到这个结果,我是真的很意外。

2011-04-18

时间: 2024-10-31 01:38:39

Hashtable 删除元素, 抛出异常 java.util.ConcurrentModificationException的相关文章

list删除操作 java.util.ConcurrentModificationException

首先大家先看一段代码: public static void main(String[] args) { List<String> listStr = new ArrayList<String>();      listStr.add("1");      listStr.add("2");      listStr.add("3");      listStr.add("4");      listS

集合循环删除问题-报错java.util.ConcurrentModificationException解析

java.util.ConcurrentModificationException 异常问题详解 环境:JDK 1.8.0_111 在Java开发过程中,使用iterator遍历集合的同时对集合进行修改就会出现java.util.ConcurrentModificationException异常,本文就以ArrayList为例去理解和解决这种异常. 一.单线程情况下问题分析及解决方案 1.1 问题复现 先上一段抛异常的代码. 1 public void test1() { 2 ArrayList

java.util.ConcurrentModificationException 异常解决办法及原理

最近在修程序的bug,发现后台抛出以下异常: Exception in thread "main" java.util.ConcurrentModificationException at java.util.HashMap$HashIterator.nextEntry(HashMap.java:793) at java.util.HashMap$KeyIterator.next(HashMap.java:828) at com.keyman.demo.test.ClearResult

java.util.ConcurrentModificationException异常分析

Java在操作ArrayList.HashMap.TreeMap等容器类时,遇到了java.util.ConcurrentModificationException异常.以ArrayList为例,如下面的代码片段: [java] view plaincopy import java.util.ArrayList; import java.util.Iterator; import java.util.List; import java.util.concurrent.CopyOnWriteArr

java集合--java.util.ConcurrentModificationException异常

ConcurrentModificationException 异常:并发修改异常,当方法检测到对象的并发修改,但不允许这种修改时,抛出此异常.一个线程对collection集合迭代,另一个线程对Collection进行修改的时候, 就会出现上面的异常. 下面看一下代码: package cn.itcast.p4.list.demo; import java.util.ArrayList; import java.util.Iterator; import java.util.List; pub

java.util.ConcurrentModificationException的解决办法

今天在使用iterator.hasNext()操作迭代器的时候,当迭代的对象发生改变,比如插入了新数据,或者有数据被删除. 编译器报出了以下异常: Exception in thread "main" java.util.ConcurrentModificationException at java.util.LinkedHashMap$LinkedHashIterator.nextNode(LinkedHashMap.java:719) at java.util.LinkedHash

android详细信息java.util.ConcurrentModificationException变态

在今天做android当项目,我遇到了这个异常,好吧.其实最不寻常遇到异常IllegalstateException.它们与我们的硬件连接SDK抛出,我想折磨学生阿玉啊.扯远了. 今天,我想回到这个异常.java.util.ConcurrentModificationException异常,一開始我愣了一下.貌似从来没遇到过这个,然后果断百度大神.这才发现: 原因是你遍历该集合时.对该集合进行了删除元素的操作导致的.假设你有删除元素的必要,建议赋值到还有一个集合,然后对他进行删除操作. 偶出现错

小小异常:java.util.ConcurrentModificationException

1 public static void main(String[] args) { 2 ArrayList<String> list = new ArrayList<String>(); 3 for(int i = 0; i < 5; i++){ 4 list.add("str"+i); 5 } 6 for(String item:list){ 7 if("str2".equals(item)){ 8 list.remove(&quo

java.util.ConcurrentModificationException 解决办法

在使用iterator.hasNext()操作迭代器的时候,如果此时迭代的对象发生改变,比如插入了新数据,或者有数据被删除. 则使用会报以下异常: java.util.ConcurrentModificationException         at java.util.HashMap$HashIterator.nextEntry(HashMap.java:793)         at java.util.HashMap$KeyIterator.next(HashMap.java:828)