登录
首页 >  文章 >  java教程

Java遍历Set的几种方法

时间:2026-03-06 11:25:31 105浏览 收藏

本文深入解析了Java中遍历Set集合的四种主流方式——增强for循环、Iterator手动遍历、Stream函数式操作和toArray()转数组,强调增强for循环是语法最简洁、安全性最高的首选方案;特别指出遍历时如需删除元素,必须使用Iterator.remove()而非集合自身的remove()方法,否则将触发ConcurrentModificationException;同时澄清Stream适用于一次性过滤映射等函数式场景但不修改原集合,而toArray()仅在对接遗留系统等极少数兼容性需求下才值得考虑;最后提醒开发者:HashSet本身无序是设计使然,遍历顺序不确定并非bug,而是其核心契约。

在Java中如何遍历Set集合_Java集合遍历方式解析

用增强for循环遍历Set最常用也最安全

Java中Set本身不保证顺序(除非是LinkedHashSetTreeSet),但遍历逻辑和List一致。增强for循环是首选,它底层调用Iterator,自动处理并发修改检查,且语法简洁:

Set<String> set = new HashSet<>();
set.add("a"); set.add("b");
for (String s : set) {
    System.out.println(s); // 输出顺序不确定
}

注意:不能在循环体中直接调用set.remove(),否则抛ConcurrentModificationException;如需边遍历边删,必须用Iterator.remove()

用Iterator手动遍历才能安全删除元素

当需要在遍历过程中动态移除满足条件的元素时,必须显式获取Iterator并调用其remove()方法——这是唯一被允许的删除方式:

Iterator<String> it = set.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.startsWith("a")) {
        it.remove(); // ✅ 安全删除
    }
}
  • it.remove()只能在next()之后调用一次,重复调用会抛IllegalStateException
  • 不能用set.remove()替代,哪怕对象相等也不行
  • ConcurrentHashMap.newKeySet()这类并发集合,应改用removeIf()或专用迭代器

Stream遍历适合过滤/映射后的一次性消费

Java 8+ 可用stream()做函数式遍历,适合链式操作,但要注意它不改变原集合,且每次调用都新建流:

set.stream()
   .filter(s -> s.length() > 1)
   .map(String::toUpperCase)
   .forEach(System.out::println);

常见误区:

  • 不要在forEach()里修改原Set,这属于副作用,既不安全也不符合函数式原则
  • 若需收集结果,用collect(Collectors.toSet()),别试图复用原引用
  • parallelStream()对小集合反而慢,且HashSet无序特性会让并行结果更难预测

toArray()转数组遍历只在特定场景有用

Set转成数组再遍历,通常只为兼容老接口或需要随机访问索引。但代价是额外内存分配和失去泛型类型信息(除非用带泛型参数的重载):

// ❌ 不推荐:丢失类型,可能ClassCastException
Object[] arr = set.toArray();

// ✅ 推荐:保留类型
String[] arr2 = set.toArray(new String[0]); // JDK 11+ 可传空数组

真正需要它的场景极少,比如调用某个只接受String[]的遗留JNI方法。日常开发中,优先走IteratorStream路径。

最容易被忽略的是:HashSet的遍历顺序与插入顺序无关,哪怕两次遍历同一组数据,输出也可能不同——这不是bug,是它的设计契约。

本篇关于《Java遍历Set的几种方法》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>