因为 Collection 接口不保证有序或支持下标访问,HashSet、LinkedHashSet 等无 get(int index) 方法,故无法用传统 for 循环遍历。

为什么不能直接用 for 循环遍历所有集合?
因为 Collection 接口本身不保证元素有序或支持下标访问,像 HashSet 和 LinkedHashSet 没有 get(int index) 方法,for (int i = 0; i 会编译失败。只有实现 List 的集合(如 ArrayList)才支持这种写法,但强行依赖下标会破坏接口抽象,也不适用于 Map 的键值对遍历。
Iterator 是统一的“游标式”访问协议,所有 Collection 子类都必须提供 iterator() 方法,这才是安全、通用的遍历入口。
如何正确调用 next() 和 hasNext()?
next() 不检查是否还有元素就直接返回并移动指针;hasNext() 才是判断依据。常见错误是先 next() 再判断,导致 NoSuchElementException。
- 必须在每次
next()前调用hasNext() - 一个
next()调用只消费一个元素,不可重复调用获取同一元素 - 迭代器状态独立于原集合 —— 即使集合后续被修改,已生成的迭代器仍按创建时快照运行(但多数实现会快速失败)
Iteratorit = list.iterator(); while (it.hasNext()) { String item = it.next(); // 安全:hasNext 已确保 next 有效 System.out.println(item); }
remove() 方法只能在 next() 之后立即调用
Iterator.remove() 是唯一安全的遍历中删除方式。直接调用 list.remove(obj) 或 list.remove(index) 会导致 ConcurrentModificationException(除非使用 CopyOnWriteArrayList 等特殊集合)。
立即学习“Java免费学习笔记(深入)”;
但注意:remove() 必须紧跟在一次 next() 之后,且只能调用一次 —— 否则抛 IllegalStateException。
- 不能在首次
next()前调用remove() - 不能连续两次调用
remove()(中间无next()) - 部分集合(如
Arrays.asList()返回的集合)的迭代器不支持remove(),调用即抛UnsupportedOperationException
Iteratorit = numbers.iterator(); while (it.hasNext()) { Integer n = it.next(); if (n % 2 == 0) { it.remove(); // ✅ 正确:刚调用过 next() } }
增强 for 循环(for-each)底层就是 Iterator
for (String s : list) 在编译后等价于显式使用 Iterator,所以它同样受上述规则约束:无法在遍历时修改集合结构,也无法获取当前索引或反向遍历。
真正需要跳过、重试、或根据前一个元素做判断时,显式使用 Iterator 更清晰可控;而 for-each 仅适合纯读取场景。
另外,Map 本身不是 Collection,但它的 keySet()、values()、entrySet() 都返回 Collection,因此可直接用于 for-each 或获取 Iterator:
for (Map.Entryentry : map.entrySet()) { System.out.println(entry.getKey() + "=" + entry.getValue()); }
迭代器的“一次性”和“单向性”是设计本质,不是限制 —— 它把遍历逻辑从集合内部解耦出来,让并发、过滤、延迟计算等扩展成为可能。别试图绕过 hasNext(),也别指望 Iterator 支持 previous() 或 reset(),那是 ListIterator 的职责。










