Iterator是Java集合遍历的标准方式,通过hasNext()和next()解耦遍历逻辑与集合内部结构;遍历时需用it.remove()而非集合remove()以避免ConcurrentModificationException;增强for循环是其语法糖,ListIterator支持双向遍历和修改,Stream适用于函数式数据处理。

Java里的Iterator是集合遍历的标准方式,它把“怎么取元素”和“集合内部结构”解耦了——你不用管ArrayList是数组还是LinkedList是链表,只要拿到Iterator,就能统一用hasNext()和next()来遍历。
Iterator的基本用法:三步走
几乎所有Collection接口的实现类(如ArrayList、HashSet、LinkedList)都支持通过iterator()方法获取迭代器:
- 调用集合的
iterator()方法,得到一个Iterator对象 - 用
hasNext()判断是否还有下一个元素(返回boolean) - 有就调用
next()获取该元素(返回Object或泛型类型),同时内部指针自动后移
示例:
Listlist = Arrays.asList("a", "b", "c"); Iterator it = list.iterator(); while (it.hasNext()) { String s = it.next(); System.out.println(s); }
为什么不能边遍历边用集合的remove()?
直接调用集合自身的remove()(比如list.remove(obj))在遍历时会触发ConcurrentModificationException。因为Iterator在创建时会记录集合的修改次数(modCount),每次集合结构变化(增删元素)都会更新这个值;而Iterator自己的remove()方法会同步更新预期值,保证安全。
立即学习“Java免费学习笔记(深入)”;
正确做法是使用Iterator的remove():
- 必须先调用过
next(),否则抛IllegalStateException - 每个
next()后最多调用一次remove() - 它删除的是上一次
next()返回的那个元素
示例(删除所有"b"):
Iteratorit = list.iterator(); while (it.hasNext()) { if ("b".equals(it.next())) { it.remove(); // 安全删除 } }
增强for循环(for-each)其实是Iterator的语法糖
写for (String s : list)时,编译器会自动转成Iterator遍历。所以它也受限于Iterator的规则:
- 底层仍调用
iterator()、hasNext()、next() - 不支持在循环中修改集合结构(会报错)
- 无法获取当前索引,也不能反向遍历
如果需要索引或修改操作,老老实实用显式Iterator更清晰、更可控。
Iterator的局限与替代方案
标准Iterator只能单向、前向遍历,且不支持替换元素。如果需要更多能力,可考虑:
-
ListIterator:仅适用于List,支持双向遍历(
hasPrevious()/previous())、添加(add())、替换(set()) - forEachRemaining()(Java 8+):传入Consumer,一次性消费剩余所有元素,简洁但不可中断
- Stream API(Java 8+):支持函数式操作(filter/map/collect等),适合复杂数据处理,但注意它是惰性求值,且不改变原集合
实际选型看需求:简单遍历用增强for;要安全删除用Iterator;要双向或改内容用ListIterator;要链式处理用Stream。










