哈喽,大家好,我是了不起。
通常1-3年工作经验的程序员算是初级程序员,再往后基本上就是在编程领域有了一定经验的高级程序员了。
但是最近公司代码review时,我居然发现一个 5 年工作经验的程序员,使用 ArrayList 居然用 forEach 遍历删除元素?
1、现场还原
由于公司代码有一定敏感,我这里把代码进行脱敏,大家一起来看看:
1 |
|
乍看之下,这段代码似乎没什么问题。但实际运行时,它会抛出ConcurrentModificationException
异常。
这是为什么呢?我们运行这段代码,报错如下 :
2、原因分析
其实 forEach 是一个语法糖,我们编译后的代码如下:
1 |
|
然后这里的 i.next() 方法:
1 |
|
这样就很明了了,在Java中,当我们试图在遍历一个集合的同时修改它时,就会遇到ConcurrentModificationException
。这是因为ArrayList的迭代器设计为快速失败(fail-fast),即在检测到集合在迭代期间被修改时立即抛出异常。
3、如何正确删除?
3.1 使用迭代器的remove方法
1 |
|
这种方法可以保证在删除元素的同时不会破坏迭代器的状态。
3.2 使用removeIf方法
从Java 8开始,ArrayList引入了removeIf方法,这是删除元素的另一种便捷方式:
1 |
|
3.3 收集需要删除的元素
最后一种方法是首先收集所有需要删除的元素,然后再进行删除:
1 |
|