for-each循环最安全常用,适用于只读遍历;需删除元素时必须用Iterator.remove();传统for索引循环仅适用于List且注意性能差异;Lambda forEach适合简单操作,复杂逻辑应避免。

用 for-each 循环遍历 Collection 最安全也最常用
绝大多数情况下,for-each(增强型 for)是首选:它简洁、不易出错、自动处理迭代器逻辑。适用于所有实现 Iterable 的集合(如 ArrayList、HashSet、LinkedList)。
注意它底层调用的是 iterator(),所以遍历时**不能直接调用 collection.remove()**,否则抛 ConcurrentModificationException。
for (String item : list) {
System.out.println(item);
// ❌ 不能写 list.remove(item); —— 会触发 fail-fast
}
- 适合只读遍历或已知不修改集合结构的场景
- 无法获取当前索引,也不支持反向遍历
- 对
Map需先转成entrySet()、keySet()或values()
用 Iterator 手动控制遍历并安全删除元素
当需要在遍历中动态删除匹配项时,必须用 Iterator 的 remove() 方法——这是唯一被允许的删除方式。
Iteratorit = list.iterator(); while (it.hasNext()) { String item = it.next(); if ("target".equals(item)) { it.remove(); // ✅ 安全删除 } }
-
it.remove()只能紧跟在it.next()之后调用一次 - 不能用于数组或基本类型数组;仅适用于
Collection子类 - 部分并发集合(如
CopyOnWriteArrayList)的Iterator不支持remove()
用传统 for 索引循环只适用于 List 实现
get(i) 是 List 接口定义的方法,所以只有 ArrayList、LinkedList 等才支持下标访问。但性能差异很大:
立即学习“Java免费学习笔记(深入)”;
-
ArrayList.get(i)是 O(1),适合随机访问 -
LinkedList.get(i)是 O(n),每次都要从头/尾跳转,遍历时应避免
for (int i = 0; i < list.size(); i++) {
String item = list.get(i); // ✅ 对 ArrayList 高效;❌ 对 LinkedList 效率低
System.out.println(item);
}
另外,若在循环中增删元素,size() 和索引容易不同步,极易漏项或越界。
forEach() + Lambda 适合简单操作,但有作用域限制
JDK 8 引入的 Collection.forEach(Consumer) 写法简洁,但本质仍是内部调用 Iterator,因此同样不支持遍历时修改集合。
list.forEach(item -> {
System.out.println(item);
// ❌ 不能写 list.remove(item);
});
- Lambda 表达式里只能访问
final或“事实上 final”的变量 - 无法使用
break或continue控制流程(需改用传统循环) - 异常处理麻烦:Lambda 不能抛受检异常,必须包装成运行时异常
真正复杂的逻辑(比如带状态计数、条件中断、嵌套操作),别硬套 forEach,老实用 Iterator 或传统循环。










