登录
首页 >  文章 >  java教程

JavaMap遍历方法与键值操作技巧

时间:2026-01-05 10:25:03 147浏览 收藏

今日不肯埋头,明日何以抬头!每日一句努力自己的话哈哈~哈喽,今天我将给大家带来一篇《Java Map遍历方法及键值技巧》,主要内容是讲解等等,感兴趣的朋友可以收藏或者有更好的建议在评论提出,我都会认真看的!大家一起进步,一起学习!

最推荐使用 for (Map.Entry entry : map.entrySet()) 遍历,兼顾可读性、性能(避免重复哈希查找)和兼容性(Java 5+),但需判空、写全泛型、禁止遍历时结构修改。

Java里Map遍历有哪些常见方式_Java键值遍历技巧说明

最常用、最推荐的方式是 for (Map.Entry entry : map.entrySet()) —— 它兼顾可读性、性能和 JDK 兼容性(Java 5+),且不会额外触发 map.get() 查找。

entrySet() 遍历键值对(推荐首选)

这是绝大多数场景下的最优解:一次获取键和值,避免重复哈希查找。底层直接访问内部 Node 数组,不调用 get(),性能比 keySet() + get() 高约 10%~15%,尤其在大 Map 中更明显。

  • 必须检查空引用:if (map != null && !map.isEmpty()),否则 map.entrySet() 在 null 上会抛 NullPointerException
  • 泛型务必写全,如 Map.Entry,避免运行时类型擦除导致的 ClassCastException
  • 不要在遍历中修改 Map 结构(如 put()remove()),否则会触发 ConcurrentModificationException;如需删除,请用 Iterator.remove()
Map<String, Integer> map = new HashMap<>();
map.put("a", 1);
map.put("b", 2);
<p>for (Map.Entry<String, Integer> entry : map.entrySet()) {
System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}</p>

只遍历 key 或 value 时用 keySet()values()

当你明确只需要键或只处理值(比如统计所有分数、收集所有用户名),用 keySet()values() 更简洁、稍快,且语义更清晰。

  • keySet() 遍历时若还需 value,**必须调用 map.get(key)** —— 这会多一次哈希计算和可能的链表/红黑树遍历,小 Map 不明显,但万级数据下可观测到延迟上升
  • values() 返回的是 Collection,不保证顺序(除非用 LinkedHashMap),也不能反查 key
  • 二者都不支持修改原 Map(如 remove() 会抛 UnsupportedOperationException),想安全删元素请回到 entrySet().iterator()
// 只要 key
for (String key : map.keySet()) {
    System.out.println("Key: " + key);
}
<p>// 只要 value
for (Integer value : map.values()) {
System.out.println("Value: " + value);
}</p>

需要控制遍历流程或兼容老 JDK 时用 Iterator

当你要在遍历中途 break、continue,或需在 Android(旧版 ART)、Java 6/7 环境下运行,Iterator 是唯一可控、兼容性最强的方式。它也允许安全删除元素。

  • 务必用 entrySet().iterator(),而不是 keySet().iterator() 再去 get() —— 后者仍是两趟操作
  • 删除必须调用 iterator.remove(),不能调用 map.remove(),否则立刻抛异常
  • 泛型不写(裸类型)虽能跑,但极易引发类型错误,不建议在现代项目中使用
Iterator<Map.Entry<String, Integer>> it = map.entrySet().iterator();
while (it.hasNext()) {
    Map.Entry<String, Integer> entry = it.next();
    if (entry.getValue() > 1) {
        it.remove(); // 安全删除
    }
}

Java 8+ 场景:用 forEach()stream() 做函数式处理

适合做过滤、映射、聚合等声明式操作,代码更紧凑,但要注意:它不是“更快”,而是“更表达意图”。实际性能略低于传统 for 循环(有 Lambda 创建开销、Stream pipeline 构建成本)。

  • forEach() 是终端操作,不可链式继续;stream() 可组合 filter()map()collect(),但注意 parallelStream() 对 Map 遍历无意义(HashMap 本身无序且非分段)
  • 所有 Stream 操作默认不保证顺序(HashMap 无序),如需有序请用 LinkedHashMap 或显式 sorted()
  • 别在 forEach() 里修改外部变量(非 final 或 effectively final),编译会报错
map.forEach((key, value) -> {
    System.out.println("Key: " + key + ", Value: " + value);
});
<p>// 过滤出 value > 1 的项并转成 list
List<Map.Entry<String, Integer>> filtered = map.entrySet().stream()
.filter(entry -> entry.getValue() > 1)
.collect(Collectors.toList());</p>

真正容易被忽略的点是:遍历前是否判空、遍历中是否误删结构、以及不同方式对 Map 实现类(如 TreeMap vs HashMap)的顺序影响——entrySet()keySet()TreeMap 中天然有序,在 HashMap 中则完全无序,这点不看文档很容易踩坑。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>