使用keySet遍历Map键的技巧
时间:2026-04-14 15:59:34 138浏览 收藏
本文深入解析了Java中通过keySet()遍历Map键的正确用法与常见陷阱:明确指出keySet()返回的是不可重复、无序(除非使用LinkedHashMap或TreeMap)的Set视图,而非List或数组,因此不能下标访问或强制类型转换;重点推荐增强for循环作为最简洁安全的遍历方式,同时详解了迭代器remove()和removeIf()在安全删除场景下的必要性,并警示直接在Stream forEach中修改Map会导致并发异常;还提醒开发者注意keySet的“视图”本质——零内存开销但生命周期依赖原Map,以及entrySet遍历在性能敏感场景下的优势,帮助读者避开ClassCastException、ConcurrentModificationException等典型错误,写出更健壮高效的Map操作代码。

keySet() 返回的是 Set 还是 List?
keySet() 方法返回的是一个 Set,不是 List,更不是数组。这意味着它天然不保证顺序(除非你用的是 LinkedHashMap 或 TreeMap),且不允许重复键——这正好符合 Map 的语义。直接对返回值调用 get(0) 或尝试下标访问会编译失败,因为 Set 没有 get(int) 方法。
常见错误现象:java.lang.ClassCastException: java.util.HashMap$KeyIterator cannot be cast to java.util.List —— 有人试图把 keySet() 强转成 List 后再操作,这是错的。
- 要用遍历,就走标准集合遍历路径:增强 for、迭代器或 Stream
- 真需要索引访问?先转成
ArrayList:new ArrayList(map.keySet()),但注意这会产生额外对象和复制开销 TreeMap的keySet()是有序的(自然序或自定义比较器);LinkedHashMap的是插入序;HashMap的是不确定的
用增强 for 遍历 keySet 最简洁
这是最常用也最安全的方式,代码清晰、无空指针风险(前提是 map 本身非 null),且 JVM 对其做了充分优化。
Map<String, Integer> map = new HashMap<>();
map.put("a", 1);
map.put("b", 2);
for (String key : map.keySet()) {
System.out.println(key + " → " + map.get(key));
}
注意点:
- 不要在循环体内调用
map.remove(key),否则抛ConcurrentModificationException - 如果只是想检查某个键是否存在,直接用
map.containsKey(key),别先取keySet()再遍历查找 - 如果后续还要频繁查 value,建议直接遍历
entrySet(),避免重复哈希查找——map.get(key)在大 Map 中可能成为性能瓶颈
用迭代器遍历时如何安全删除键
当需要根据条件删掉某些键时,必须用迭代器的 remove() 方法,不能用 map.remove()。
Iterator<String> iter = map.keySet().iterator();
while (iter.hasNext()) {
String key = iter.next();
if (key.startsWith("temp")) {
iter.remove(); // ✅ 安全
// map.remove(key); // ❌ 会触发 ConcurrentModificationException
}
}
原因:keySet() 返回的是 Map 的“视图”,它的迭代器与底层 Map 共享结构状态。只有迭代器自己的 remove() 会同步更新内部 modCount。
- 用
removeIf()更简练:map.keySet().removeIf(key -> key.startsWith("temp")) - 该方法内部也是基于迭代器实现,语义明确且线程不安全(同原始 Map)
- 别对
keySet()调用stream().forEach(...)并在里面删 key,同样会出错
Stream 遍历 keySet 的适用场景和坑
适合做函数式转换、过滤、收集,比如提取所有以某前缀开头的键并转成小写列表:
List<String> filteredKeys = map.keySet().stream()
.filter(k -> k.startsWith("A"))
.map(String::toLowerCase)
.collect(Collectors.toList());
但要注意:
- 每次调用
keySet().stream()都会新建一个 Stream,不会复用;频繁调用要考虑是否真需要 Stream 抽象 - Stream 不是为修改设计的,
forEach(System.out::println)可以,但forEach(map::remove)会报错(非并发安全的修改) - 如果 map 很大,
keySet()本身不占额外内存(它是视图),但stream().collect(...)会生成新集合,注意 GC 压力
复杂点在于:keySet 视图的生命周期完全绑定原 Map,一旦 map 被清空或重建,旧的 keySet 引用就变成“空壳”,但不会立即失效——它仍可遍历,只是返回空结果。这点容易被忽略。
本篇关于《使用keySet遍历Map键的技巧》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
443 收藏
-
475 收藏
-
104 收藏
-
163 收藏
-
467 收藏
-
133 收藏
-
178 收藏
-
175 收藏
-
339 收藏
-
288 收藏
-
404 收藏
-
140 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习