使用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 对其做了充分优化。
Mapmap = 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()。
Iteratoriter = 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 的适用场景和坑
适合做函数式转换、过滤、收集,比如提取所有以某前缀开头的键并转成小写列表:
ListfilteredKeys = 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 收藏
-
文章 · java教程 | 16小时前 | Spring Boot · Java教程 · 接口设计 · Webhook · 幂等设计 · java spring boot WebHook 回调接口 幂等 状态流转 验签488 收藏
-
文章 · java教程 | 2天前 | Java教程 · TTL缓存 · ConcurrentHashMap · 小项目 · java 本地缓存 concurrenthashmap TTL缓存 过期淘汰394 收藏
-
355 收藏
-
495 收藏
-
365 收藏
-
455 收藏
-
文章 · java教程 | 1星期前 | hashmap · 集合 · Java教程 · hashCode · equals · java HashMap map equals hashCode 可变key474 收藏
-
178 收藏
-
文章 · java教程 | 1星期前 | map · 并发安全 · 缓存设计 · Java教程 · java optional concurrenthashmap computeIfAbsent Map缓存236 收藏
-
204 收藏
-
文章 · java教程 | 2星期前 | Java · 集合 · ArrayList · Iterator · removeIf · java iterator ArrayList ConcurrentModificationException removeIf410 收藏
-
文章 · java教程 | 2星期前 | Java · 异步编程 · 后端开发 · CompletableFuture · 接口聚合 · java 结果合并 completablefuture 并行调用 超时兜底428 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习