java.util.ConcurrentModificationException

集合类并发出现juc并发修改异常问题:java.util.ConcurrentModificationException

public class ErrorArray {
    public static void main(String[] args) {
//        List<String> list = Arrays.asList("a", "b", "c");
//        list.forEach(System.out::println);

//        抛出java.util.ConcurrentModificationException异常
        List<String> stringArrayList = new CopyOnWriteArrayList<>();
        for (int i=0;i<=30;i++) {
            new Thread(() -> {
                stringArrayList.add(UUID.randomUUID().toString().substring(0, 8));
                System.out.println(stringArrayList);
            },String.valueOf(i)).start();
        }
/*
* 故障现象:
*        java.util.ConcurrentModificationException
* 导致原因:
*       多线程并发修改出错,一个正在写入,另一个过来抢夺,导致数据不一致异常。并发修改异常
*       list的add方法并不是线程安全的
* 解决方案:
          1:使用new vector<>();
          2:使用Collections.synchronizedList(new ArrayList<>());
          3:使用juc的new CopyOnWriteArrayList<>();
* 优化建议(同样的错误不犯第二次):
         写时复制
         CopyonWrite容器即写时复制的容器。往一个 容器添加元素的时候,不直接往当前容器object[]添加,而是先将当前容器object[ ]进行Copy, .
         复制出一-个新的容器0bject[] newElements, 然后新的容器object[] newElements 里添加元素,添加完元素之后,
         再将原容器的引用指向新的容器setArray(newElements);。 这样做的好处是可以对CopyOnWrite容器进行并发的读,
         而不需要加锁,因为当前容器不会添加任何元素。所以CopyOnWrite 容器也是一种读写分离的思想,读和写不同的容器
         public boolean add(E e)
         final ReentrantLock lock = this. lock;
         lock. lock();
         try{
         object[] ele ments = getArray();
         int Len = elements. length;
         0bject[] newElements = Arrays. copyof(elements, len + 1);
         newELements[len] = e;
         setArray(newELements);
         return true;
         finally {
         lock. unlock();
         }
         }
* */


    }
}

HashSet也有类似的现象,处理的方式也类似:

public class ErrorSet {
    public static void main(String[] args) {
//        List<String> list = Arrays.asList("a", "b", "c");
//        list.forEach(System.out::println);

//        抛出java.util.ConcurrentModificationException异常
        Set<String> list =Collections.synchronizedSet(new HashSet<>());
        for (int i=0;i<=30;i++) {
            new Thread(() -> {
                list.add(UUID.randomUUID().toString().substring(0, 8));
                System.out.println(list);
            },String.valueOf(i)).start();
        }
/*
* 故障现象:
*        java.util.ConcurrentModificationException
* 导致原因:
*       多线程并发修改出错,一个正在写入,另一个过来抢夺,导致数据不一致异常。并发修改异常
*       list的add方法并不是线程安全的
* 解决方案:
          1:使用juc的new CopyOnWriteArraySet<>();
          2:使用Collections.synchronizedSet(new HashSet<>());
 */


    }
}

hashSet的底层结构是hashMap,
在这里插入图片描述
之所以使用add添加元素是一个值,是因为value在内部已经封装了,是一个空对象。

在这里插入图片描述

在这里插入图片描述

HashMap也有类似的现象,处理的方式也类似:

public class ErrorArray {
    public static void main(String[] args) {
//        List<String> list = Arrays.asList("a", "b", "c");
//        list.forEach(System.out::println);

//        抛出java.util.ConcurrentModificationException异常
       Map<String,String> map= Collections.synchronizedMap(new HashMap<>());
        for (int i=0;i<=30;i++) {
            new Thread(() -> {
                map.put(Thread.currentThread().getName(),UUID.randomUUID().toString().substring(0, 8));
                System.out.println(map);
            },String.valueOf(i)).start();
        }
/*
* 故障现象:
*        java.util.ConcurrentModificationException
* 导致原因:
*       多线程并发修改出错,一个正在写入,另一个过来抢夺,导致数据不一致异常。并发修改异常
*       list的add方法并不是线程安全的
* 解决方案:
          1:使用juc的new ConcurrentHashMap();
          2:使用 Collections.synchronizedMap(new HashMap<>());
 */


    }
}

Java.util.ConcurrentModificationException是Java编程语言中的一种异常,它通常在并发修改集合时抛出。当使用iterator.hasNext()操作迭代器时,如果在迭代过程中集合发生了改变(如插入或删除元素),就会抛出该异常。该异常的出现原因是由于集合在迭代过程中被修改,导致迭代器检测到并抛出异常。 解决Java.util.ConcurrentModificationException的方法可以通过以下几种途径: 1. 使用Iterator的remove()方法来删除元素,而不是使用集合自身的remove()方法。这是因为Iterator的remove()方法是安全的,可以避免并发修改异常的发生。 2. 使用synchronized关键字或其他同步机制来保证对集合的访问是线程安全的,避免在迭代过程中发生并发修改。 3. 使用并发集合类,如ConcurrentHashMap或CopyOnWriteArrayList,这些类内部使用了一些机制来保证并发修改的安全性,从而避免了ConcurrentModificationException的发生。 总之,要避免Java.util.ConcurrentModificationException异常的发生,需要注意在迭代集合时不要在迭代过程中对集合进行修改,并采取适当的同步机制或使用线程安全的集合类来确保并发修改的安全性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [java.util.ConcurrentModificationException 解决方法](https://download.csdn.net/download/weixin_38723192/14093201)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [java.util.ConcurrentModificationException: null 报错解决](https://blog.csdn.net/qq_51741039/article/details/126008727)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值