zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    Yii2中如何使用CodeCeption
    切换composer国内镜像 Laravel China停用,切换阿里云composer全量镜像
    mysql 不同库不同表字段数据复制
    DataGridView绑定数据库,取得的数据插入到DataGridView指定列(一)
    Winform在一个窗体获取其他窗体的值
    winform按钮和子按钮
    C#生成唯一的ID保存到数据库
    C#Winform从页面获取数据,传入数据库
    C#winform省市县联动,以及有的县是空值时显示异常的处理
    VS2010连接SQLite数据库
  • 原文地址:https://www.cnblogs.com/personnel/p/4583045.html
Copyright © 2011-2022 走看看