集合或映射迭代过程进行删除或修改操作的时候会导致并发异常

简介: 集合或映射迭代过程进行删除或修改操作的时候会导致并发异常

一、集合for-each 边迭代边删除,导致并发修改异常【for-each和Iterator的选择】


 

❀ 结论: 当需要边迭代集合元素,边删除指定的元素时,只能使用迭代器,而且是只能使用迭代器对象的remove方法。

 

 

1,看代码(报错代码):


5.png


2、看代码(正确代码):


6.png


3,解释一下for-each迭代时需要删除,为什么不能选择集合的方法来删除,而是选择迭代器的方法删除?

1):因为会出现并发修改异常ConcurrentModificationException】----并发(说明至少是两个线程---等下立马解释这两个线程如何导致报错的哈)

2):for-each 底层:就是使用迭代器啦

[ 验证:通过for-each的例子中生成class文件使用反编译工具,反编译结果出现了java.util.Iterator类,并且出现了其hasNext和next方法。(补充一下:不同反编译工具出来的结果可能不太一样,像我使用的工具是:jd-gui-windows-1.6.6,就是没看到哈哈哈)]

 

4, 迭代器导致并发修改异常的原因分析:

1):并发修改异常是:两个线程mian线程和迭代器线程不同步导致的迭代集合时,除了当前的线程的执行外,生成了一个迭代器线程

 

2):for-each 迭代元素时是将main线程中的元素拷贝到迭代器线程。然后main线程继续向下执行[迭代],而迭代器的线程也是向下执行,且执行到使用集合来删除的语句list.remove(Object o)时,main 线程中元素删除,元素数量减少一个,但是迭代器的元素没有减少【即 Collection 接口存在的删除指定元素的方法 boolean remove(Object ele),只能删除集合中的元素,却不能删除迭代器中指定的元素。

 

3)而迭代器内部在每次继续向下迭代(next方法或remove方法)时,有一个检查方法checkForComodification

    迭代器线程每次都会将和当前线程(main线程)中的元素是否个数相同,若是不同,则报错:ConcurrentModificationException。

 


6.png


■ 图是:两个线程mian线程和迭代器线程


7.png


二、映射迭代过程进行删除或修改操作的时候会导致并发异常


7.png



✿ 映射报错的解决:

看需求,若是咱想要迭代的映射的key值,再创建一个keySet,然后使用新创建的keySet 做迭代,而不是直接使用 map.keySet() 做迭代。(想要迭代映射的value值是同理的)


8.png



目录
相关文章
|
6月前
|
消息中间件 存储 Serverless
函数计算产品使用问题之想要请求持久化该怎么操作
函数计算产品作为一种事件驱动的全托管计算服务,让用户能够专注于业务逻辑的编写,而无需关心底层服务器的管理与运维。你可以有效地利用函数计算产品来支撑各类应用场景,从简单的数据处理到复杂的业务逻辑,实现快速、高效、低成本的云上部署与运维。以下是一些关于使用函数计算产品的合集和要点,帮助你更好地理解和应用这一服务。
|
6月前
new 一个对象的过程中发生了什么
new 一个对象的过程中发生了什么
|
7月前
|
NoSQL MongoDB 数据库
通过优化索引以消除 MongoDB 中的 "查询目标已超过1000个扫描对象/返回的文档数" 警告
MongoDB NoSQL数据库在处理复杂查询时可能出现“查询目标已超过1000个扫描对象/返回的文档数”警告。文章分析了该问题,展示了一个示例集合和相关索引,并提供了查询示例。通过`explain`命令发现查询未有效利用索引。解决方案是遵循ESR规则,创建新索引从而优化查询并消除警告。
193 1
|
7月前
怎么确保一个集合不能被修改?
怎么确保一个集合不能被修改?
|
7月前
|
缓存 架构师 NoSQL
五种更新缓存的组合方式
【4月更文挑战第19天】更新缓存的步骤特别简单,共两步:更新数据库和更新缓存。但这简单的两步中需要考虑很多问题。
每日一道面试题之怎么确保一个集合不能被修改?
每日一道面试题之怎么确保一个集合不能被修改?
|
消息中间件 数据采集 Kafka
每次join之后没有正确处理数据的重复或缺失情况
每次join之后没有正确处理数据的重复或缺失情况
129 1
|
算法
【递归可以干什么】1#重复执行某种模式
【递归可以干什么】1#重复执行某种模式
87 0
怎么确保一个集合不能被修改
怎么确保一个集合不能被修改
下一篇
DataWorks