关于ArrayList与CopyOnWriteArrayList的遍历remove操作

今天在测试list的遍历删除的时候突然发现,有报错

1、当List中有五个对象的时候,使用CopyOnWriteArrayList实例化

    

    

    我们会发现当使用foreach方法删除的时候是没有问题的,但是会发现,当使用迭代方法 iterator.remove() 删除的时候就会报错

    

2、使用ArrayList实例化对象

    

    

    这个和我们第一个的结果正好相反,使用Iterator这个方法是可以直接删除的,而使用foreach方法是会报错

    

    

3、首先分析Iterator这个方法

    我们可以看到第一种返回的是一个UnsupportedOperationException这个错,说明是不支持该操作的,那么我们就需要去看看CopyOnWriterArrayList的源代码

    当你直接跳转到CopyOnWriteArrayList中你会发现CopyOnWriteArrayList并没有实现Iterator,但是我们可以在下方发现

        

    有内部类实现了ListIterator,而ListIterator是实现了Iterator的,所以,我们在迭代的时候是直接跳转到这个内部类中的,然后我们会发现在这个内部类中有一个remove方法,直接返回了UnsupportedOperationException,也就是最起码1.8的jdk是不支持CopyOnWriteArrayList进行迭代删除的

        

     但是我们再看看ArrayList的源码的时候会发现,其中有一个内部类直接实现了Iterator

        

    而且其中重写了remove方法,当每次删除操作之后,会重新将modCount赋值给expectedModCount,这样的话,当我们每次remove之前调用checkForComdification这个方法的时候,就不会抛出第二种情况的异常了,也就是ConcurrentModificationException

        

4、分析foreach这个方法,也就是list.remove()

    我们可以先看一下ArrayList的错误,我们会发现,我们并没有使用到Itr,但是错误中首先是用的Itr中的checkForComodification这个方法

        

    在看了其他大佬的博客之后我发现,原来foreach其实是相当于

        

    那么大家就能理解了吧,当使用ArrayList进行remove操作的时候也是会进行checkForComodification判断modCount以及expectedModCount的数值的,但是因为当我们删除完成之后,并没有同步两个数据,所以会报错,除非我们自己重写方法,将数据同步,这样就可以了

    然后我们在看看CopyOnWriteArrayList的remove方法

       

    我们会发现,系统直接生成新数组newElements,而且,因为CopyOnWriteArrayList并没有Iterator.remove这个方法,所以并不会判断修改次数与预计修改次数是否相同,所以不会报错

    关于CopyOnWriteArrayList的优缺点我会在继续研究的

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: 好的,下面是关于ArrayList的一些信息: ArrayList是Java中的一个动态数组类,它实现了List接口,能够动态地添加、删除元素,并且可以自动扩展容量。ArrayList中的元素可以是任何对象,包括null。 在ArrayList中,元素的位置是由它们在数组中的索引决定的。它支持随机访问,也就是说可以通过索引值来获取或更新元素。此外,ArrayList还提供了一些便捷的方法,如add、remove和size等。 需要注意的是,ArrayList不是线程安全的,如果多个线程同时对它进行操作,可能会导致不可预测的结果。如果需要在多线程环境中使用动态数组,可以考虑使用Vector类或者在对ArrayList进行操作时进行同步处理。 ### 回答2: ArrayList是Java中的一种数据结构,它可以动态地存储和管理一组对象。 ArrayList具有以下特点: 1. 动态大小:与传统的数组不同,ArrayList的大小是可以动态调整的。当需要添加新元素时,ArrayList会自动扩展其容量。同样地,当需要删除元素时,ArrayList会自动收缩其容量。 2. 随机访问:ArrayList中的元素可以通过索引进行随机访问。这意味着我们可以通过指定索引的方式快速地访问或更新特定位置的元素。 3. 具有重复元素:ArrayList中可以包含重复的元素。这意味着我们可以向ArrayList中添加相同的元素多次。 4. 支持迭代:ArrayList实现了Java的Iterable接口,因此我们可以使用迭代器(Iterator)来遍历ArrayList中的元素。 5. 自动装箱/拆箱:ArrayList可以自动处理基本数据类型和其对应的包装类型之间的转换。这意味着我们可以将int、char等基本数据类型直接添加到ArrayList中,而不需要手动进行转换。 6. 线程不安全:ArrayList是非线程安全的。这意味着当多个线程同时访问和修改ArrayList时,可能会发生竞态条件和数据不一致的问题。在多线程环境下,应该使用线程安全的集合类,如Vector或CopyOnWriteArrayList。 尽管ArrayList有很多优点,但也有一些缺点。例如,在删除或插入元素时,需要移动其他元素,这可能会导致性能下降。此外,ArrayList的大小只能受限于可用内存大小。 总之,ArrayList是一个非常有用的数据结构,适用于需要频繁对元素进行添加、删除、修改和随机访问的场景。同时,我们也应该根据具体的需求和性能要求选择合适的集合类。 ### 回答3: ArrayList是Java中常用的数据结构,它是基于数组实现的动态数组。它能够自动调整大小,可以存储任意类型的对象。 ArrayList具有以下特点: 1. 可变长度:与数组不同,ArrayList的长度可以动态增长或缩小。当元素超出当前容量时,会自动进行扩容。这种自动调整大小的功能使得ArrayList非常灵活,适合处理大小不确定的数据集合。 2. 快速访问:ArrayList通过下标索引来访问元素,因此可以快速地进行读取和修改操作。相比链表等其他数据结构,ArrayList的访问速度更快。 3. 有序存储:ArrayList按照元素的插入顺序进行存储,且可以保持元素的有序性。这样,可以方便地进行元素的查找和排序等操作。 4. 允许重复元素:ArrayList允许存储重复的元素。当需要统计某个元素的出现次数或保留所有重复元素时,ArrayList是一个很好的选择。 5. 内存占用:由于ArrayList是动态数组,会预留一些额外的空间用于扩容。因此,如果元素数量明确,或者预知某个ArrayList不会太大,可以使用指定初始容量来减少内存占用。 总之,ArrayList是Java中非常常用的数据结构,它具有动态大小、快速访问、有序存储和允许重复元素等特点。在编写Java程序时,我们可以根据实际需求选择合适的数据结构,包括ArrayList,来方便地进行数据的存储、操作和访问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值