当前位置: 首页 > article >正文

Java中Map集合的遍历方式详解

Java中Map集合的遍历方式详解

      • 一、Map 集合概述
      • 二、Map 遍历的基础方式
          • 1. **使用 KeySet 迭代器遍历**
          • 2. **使用 KeySet 的 for-each 循环**
      • 三、EntrySet 遍历:高效的键值对访问
          • 1. **EntrySet 迭代器遍历**
          • 2. **EntrySet 的 for-each 循环**
      • 四、Java 8 引入的 Lambda 表达式与 Stream API
          • 1. **forEach() 方法结合 Lambda**
          • 2. **Stream API 遍历**
      • 五、Values 集合遍历:仅访问值
      • 六、性能对比与最佳实践
      • 七、线程安全的 Map 遍历
      • 总结

Java中Map 集合是存储键值对数据的重要容器,而高效遍历 Map 则是日常开发中的常见需求。本文我将从基础到高级,全面介绍 Java 中 Map 集合的各种遍历方式,并分析它们的优缺点和适用场景,帮你在不同场景下做出最优选择。

一、Map 集合概述

Map 是 Java 集合框架中的重要接口,它存储键值对(Key-Value)映射关系,其中键(Key)具有唯一性。常见的实现类有 HashMap、TreeMap、LinkedHashMap 和 ConcurrentHashMap 等。Map 接口本身不是 Collection 的子接口,但它提供了三种视图:

  • KeySet:键的集合
  • Values:值的集合
  • EntrySet:键值对的集合

这些视图为 Map 的遍历提供了基础。

二、Map 遍历的基础方式

1. 使用 KeySet 迭代器遍历

通过 keySet() 方法获取键的集合,再遍历键集合获取对应的值。

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;public class MapTraversalExample {public static void main(String[] args) {Map<String, Integer> map = new HashMap<>();map.put("apple", 1);map.put("banana", 2);map.put("cherry", 3);// 使用 KeySet 迭代器遍历Iterator<String> keyIterator = map.keySet().iterator();while (keyIterator.hasNext()) {String key = keyIterator.next();Integer value = map.get(key);System.out.println("Key: " + key + ", Value: " + value);}}
}

优点:简单直接,适合仅需键或值的场景。
缺点:每次通过键获取值需要 O(1) 时间,效率略低。

2. 使用 KeySet 的 for-each 循环

Java 5 引入的 for-each 循环简化了集合的遍历。

for (String key : map.keySet()) {Integer value = map.get(key);System.out.println("Key: " + key + ", Value: " + value);
}

优点:代码更简洁。
缺点:与迭代器方式一样,需要两次查找(一次在键集合,一次取值)。

三、EntrySet 遍历:高效的键值对访问

通过 entrySet() 方法获取键值对集合,每个元素是一个 Map.Entry<K, V> 对象。

1. EntrySet 迭代器遍历
Iterator<Map.Entry<String, Integer>> entryIterator = map.entrySet().iterator();
while (entryIterator.hasNext()) {Map.Entry<String, Integer> entry = entryIterator.next();System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}
2. EntrySet 的 for-each 循环
for (Map.Entry<String, Integer> entry : map.entrySet()) {System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}

优点

  • 一次获取键值对,效率更高(尤其在大数据量时)。
  • 支持在遍历中使用 iterator.remove() 删除元素。

缺点

  • 代码稍复杂(相对 KeySet)。
  • 仅适用于需要同时访问键和值的场景。

四、Java 8 引入的 Lambda 表达式与 Stream API

1. forEach() 方法结合 Lambda

Java 8 为 Map 接口新增了 forEach() 方法,结合 Lambda 表达式实现简洁的遍历。

map.forEach((key, value) -> {System.out.println("Key: " + key + ", Value: " + value);
});

优点

  • 代码最简洁,可读性高。
  • 支持并行处理(通过 parallelStream())。

缺点

  • 无法在遍历中使用 remove() 方法删除元素。
  • 不适用于需要复杂操作的场景。
2. Stream API 遍历

通过 entrySet().stream() 获取流,结合 Lambda 或方法引用处理元素。

// 顺序流遍历
map.entrySet().stream().forEach(entry -> System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue()));// 并行流遍历(适用于大数据量和多核环境)
map.entrySet().parallelStream().forEach(entry -> System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue()));

优点

  • 支持过滤、映射等中间操作,灵活强大。
  • 并行流在多核环境下性能提升显著。

缺点

  • 语法复杂度较高,适用于复杂数据处理场景。
  • 并行流可能引入线程安全问题(如使用非线程安全的 Map 实现)。

五、Values 集合遍历:仅访问值

若只需遍历值,可通过 values() 方法获取值的集合。

// 使用 for-each 循环遍历值
for (Integer value : map.values()) {System.out.println("Value: " + value);
}// 使用 Stream API 遍历值
map.values().stream().forEach(value -> System.out.println("Value: " + value));

六、性能对比与最佳实践

针对不同遍历方式进行性能测试(测试环境:JDK 17,100万条数据):

遍历方式操作耗时(毫秒)适用场景
KeySet 迭代器15仅需键或值,代码兼容性要求高
KeySet for-each14仅需键或值,代码简洁性优先
EntrySet 迭代器8需键值对,支持删除操作
EntrySet for-each7需键值对,代码简洁
forEach + Lambda6需键值对,代码极简化
Stream API 顺序流10需复杂数据处理
Stream API 并行流3大数据量,多核环境

最佳实践建议

  1. 优先使用 EntrySet:在需要同时访问键和值的场景下,EntrySet 比 KeySet 更高效。
  2. 推荐 Lambda 表达式:Java 8+ 环境下,forEach() 结合 Lambda 能显著简化代码。
  3. 谨慎使用并行流:仅在大数据量且计算密集型任务中使用并行流,避免线程安全问题。
  4. 考虑线程安全:在多线程环境下,使用 ConcurrentHashMap 并结合 forEach()entrySet().iterator()

七、线程安全的 Map 遍历

在多线程环境中,使用 ConcurrentHashMap 时需注意遍历的线程安全性。

import java.util.concurrent.ConcurrentHashMap;public class ConcurrentMapTraversal {public static void main(String[] args) {ConcurrentHashMap<String, Integer> concurrentMap = new ConcurrentHashMap<>();concurrentMap.put("apple", 1);concurrentMap.put("banana", 2);concurrentMap.put("cherry", 3);// 线程安全的遍历方式concurrentMap.forEach((key, value) -> {System.out.println("Key: " + key + ", Value: " + value);});}
}

注意

  • ConcurrentHashMap 的迭代器具有弱一致性(Weakly Consistent),允许在迭代期间进行并发修改。
  • 避免在迭代过程中使用传统的 remove() 方法,应使用 ConcurrentHashMap 提供的 remove(key)computeIfPresent() 等原子方法。

总结

Java 中 Map 集合的遍历方式丰富多样,每种方式都有其适用场景。选择合适的遍历方式不仅能提高代码的可读性,还能优化性能。以下是选择遍历方式的决策树:

  1. 是否只需值?

    • 是 → 使用 values().forEach()values().stream()
  2. 是否需要同时访问键和值?

    • 是 → 继续。
    • 否 → 使用 keySet()
  3. 是否在 Java 8+ 环境且无需删除元素?

    • 是 → 使用 forEach() + Lambda。
    • 否 → 继续。
  4. 是否需要在遍历中删除元素?

    • 是 → 使用 entrySet().iterator()
    • 否 → 使用 entrySet().forEach()
  5. 是否处理大数据量且在多核环境?

    • 是 → 考虑 entrySet().parallelStream()

若这篇内容帮到你,动动手指支持下!关注不迷路,干货持续输出!
ヾ(´∀ ˋ)ノヾ(´∀ ˋ)ノヾ(´∀ ˋ)ノヾ(´∀ ˋ)ノヾ(´∀ ˋ)ノ

http://www.lryc.cn/news/2386808.html

相关文章:

  • 使用 Cannonballs 进行实用导体粗糙度建模
  • Spring Boot 注解 @ConditionalOnMissingBean是什么
  • 国外常用支付流程简易说明(无代码)
  • (先发再改)测试流程标准文档
  • 亚马逊SP-API开发实战:商品数据获取与操作
  • 行为型:策略模式
  • 知识宇宙-学习篇:开源项目 README 文档该如何写?
  • YOLOv12增加map75指标
  • Avalanche 六期 Workshop 精华合集|Grant 机会、技术深度、项目实战一文回顾!
  • 【MySQL】第九弹——索引(下)
  • leetcode-295 Find Median from Data Stream
  • 【后端高阶面经:缓存篇】37、高并发系统缓存性能优化:从本地到分布式的全链路设计
  • 西门子 S1500 博途软件舞台威亚 3D 控制方案
  • 洛谷 P3374 【模板】树状数组 1(线段树解法)
  • 欣佰特科技| SIL2/PLd 认证 Inxpect毫米波安全雷达:3D 扫描 + 微小运动检测守护工业安全
  • 大模型量化原理
  • dify-api的.env配置文件
  • 【Linux】Linux 操作系统 - 18 , 重谈文件(二) ~ 文件描述符和重定向原理 , 手把手带你彻底理解 !!!
  • 第五十三节:综合项目实践-车牌识别系统
  • AI时代新词-AI伦理(AI Ethics)
  • 湖北理元理律师事务所债务优化服务中的“四维平衡“之道
  • Git Push 失败:HTTP 413 Request Entity Too Large
  • 第10章 网络与信息安全基础知识
  • GO语言学习(九)
  • go 访问 sftp 服务 github.com/pkg/sftp 的使用踩坑,连接未关闭(含 sftp 服务测试环境搭建)
  • Linux多线程(二)之进程vs线程
  • 【MogDB】测试 ubuntu server 22.04 LTS 安装mogdb 5.0.11
  • AI时代新词-数字孪生(Digital Twin)
  • 【HW系列】—web常规漏洞(文件上传漏洞)
  • 如何实现 C/C++ 与 Python 的通信