Java里Iterator迭代器怎么用_Java集合遍历机制解析

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或泛型类型)

    ,同时内部指针自动后移

示例:

List list = 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()方法会同步更新预期值,保证安全。

正确做法是使用Iterator的remove()

  • 必须先调用过next(),否则抛IllegalStateException
  • 每个next()后最多调用一次remove()
  • 它删除的是上一次next()返回的那个元素

示例(删除所有"b"):

Iterator it = 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。