java Map 遍历 详解

在 Java 中,Map 是一种键值对(Key-Value)形式存储数据的集合接口。遍历 Map 是日常开发中的常见需求。以下是遍历 Map 的详解,包括各种常见方式及其适用场景。


1. Map 的基本结构

在 Java 中,Map 提供了以下几种常用实现类:

  • HashMap:基于哈希表实现,元素无序,查找和插入操作效率高。
  • LinkedHashMap:有序版本的 HashMap,按插入顺序或访问顺序存储。
  • TreeMap:基于红黑树实现,按键的自然顺序或指定的比较器顺序存储。
  • ConcurrentHashMap:线程安全的哈希表实现,适用于多线程环境。

2. Map 遍历的常见方法

2.1 遍历键值对 (Key-Value Pairs)

通过 entrySet() 方法,可以获取 Map 中的键值对集合。每个键值对封装在一个 Map.Entry 对象中。

方式 1:使用 for-each 循环
import java.util.HashMap;
import java.util.Map;

public class MapIteration {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("Alice", 25);
        map.put("Bob", 30);
        map.put("Charlie", 35);

        // 遍历键值对
        for (Map.Entry<String, Integer> entry : map.entrySet()) {
            String key = entry.getKey();
            Integer value = entry.getValue();
            System.out.println(key + " = " + value);
        }
    }
}
优点:
  • 语法清晰,便于理解。
  • 最常用的方式。
输出:
Alice = 25
Bob = 30
Charlie = 35

2.2 遍历键 (Keys)

通过 keySet() 方法获取所有键的集合,然后根据键获取对应的值。

代码实现:
for (String key : map.keySet()) {
    Integer value = map.get(key);
    System.out.println(key + " = " + value);
}
优点:
  • 适用于仅对键进行操作的场景。
缺点:
  • 每次通过 map.get(key) 获取值时,可能有额外的查找开销(尤其是 HashMap)。

2.3 遍历值 (Values)

通过 values() 方法获取所有值的集合,仅用于需要操作值的场景。

代码实现:
for (Integer value : map.values()) {
    System.out.println("Value = " + value);
}
优点:
  • 直接访问值,适用于只关注值的情况。
缺点:
  • 无法获取键。

2.4 使用 Java 8 的 Stream API

Stream API 提供了更加简洁、函数式风格的遍历方式,适合在 Java 8 及以上版本中使用。

代码实现:
map.forEach((key, value) -> System.out.println(key + " = " + value));
优点:
  • 简洁高效,函数式表达更现代化。
  • 避免了显式循环代码。

2.5 遍历时修改 Map

在遍历的同时修改 Map 是一个比较特殊的需求,需要特别注意。

  • 对于普通的 HashMap,直接修改会抛出 ConcurrentModificationException
  • 可以使用 IteratorConcurrentHashMap
代码实现:
Iterator<Map.Entry<String, Integer>> iterator = map.entrySet().iterator();
while (iterator.hasNext()) {
    Map.Entry<String, Integer> entry = iterator.next();
    if (entry.getValue() < 30) {
        iterator.remove(); // 删除值小于30的键值对
    }
}
适用场景:
  • 遍历过程中需要删除或修改键值对。

3. Map 遍历方式的比较

遍历方式代码简洁性效率适用场景
entrySet() + for-each中等高效同时需要键和值的场景
keySet() + get()简单较低(有查找开销)仅需要操作键,偶尔需要获取值的场景
values() + for-each简单高效只需要操作值的场景
forEach (Stream API)简洁高效Java 8+ 环境,推荐现代化开发
Iterator较复杂中等遍历过程中需要安全删除或修改键值对的场景

4. 适配不同场景的遍历方式

4.1 同时操作键和值

最常用的场景是需要同时操作 Map 的键和值,例如打印所有键值对或对键值对进行某些操作。

  • 推荐使用 entrySet()forEach
示例:
map.forEach((key, value) -> System.out.println(key + " = " + value));

4.2 仅需要操作键

当只需要操作键时,可以直接使用 keySet()

示例:
for (String key : map.keySet()) {
    System.out.println("Key = " + key);
}

4.3 仅需要操作值

当只需要操作值时,可以直接使用 values()

示例:
for (Integer value : map.values()) {
    System.out.println("Value = " + value);
}

4.4 遍历时修改 Map

当需要在遍历的同时删除或修改键值对时,使用 Iterator 是首选。

示例:
Iterator<Map.Entry<String, Integer>> iterator = map.entrySet().iterator();
while (iterator.hasNext()) {
    Map.Entry<String, Integer> entry = iterator.next();
    if (entry.getValue() < 30) {
        iterator.remove();
    }
}

5. 特殊类型 Map 的遍历

5.1 ConcurrentHashMap

对于多线程环境下的并发操作,使用 ConcurrentHashMap 是线程安全的选择,可以直接使用 forEach 方法。

示例:
ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
map.put("Alice", 25);
map.put("Bob", 30);

map.forEach((key, value) -> System.out.println(key + " = " + value));

5.2 TreeMap

TreeMap 遍历时的顺序是根据键的自然顺序或指定的比较器顺序,可以用相同的方式遍历。


6. 总结

  1. 遍历 Map 时,优先选择 entrySet() + for-eachforEach (Java 8+)
  2. 如果只操作键或值,使用 keySet()values()
  3. 如果需要修改或删除 Map 中的元素,使用 Iterator
  4. 在多线程环境中,选择 ConcurrentHashMap

选择合适的遍历方式可以提升代码的效率和可读性,根据具体场景选择最佳方式!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/920300.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

利用 TensorFlow Profiler:在 AMD GPU 上优化 TensorFlow 模型

TensorFlow Profiler in practice: Optimizing TensorFlow models on AMD GPUs — ROCm Blogs 简介 TensorFlow Profiler 是一组旨在衡量 TensorFlow 模型执行期间资源利用率和性能的工具。它提供了关于模型如何与硬件资源交互的深入见解&#xff0c;包括执行时间和内存使用情…

二叉树——输出叶子到根节点的路径

目录 代码 算法思想 例子 思维拓展 代码 int LeaveBit(Bitree T,int flag,int g) {if (!T) {return 0;}if (T->rchild NULL && T->lchild NULL) {//cout << "empty:" << T->data << endl;s.push(T->data);while (!s.emp…

PIL学习---彩色RGB图像按通道输出

要将 RGB 图像拆分为单独的 R、G、B 通道并分别展示&#xff0c;可以通过 PIL 中的 split() 方法将图像的三个通道分开&#xff0c;并使用 matplotlib 来显示每个通道的图像。效果如下图所示&#xff1a; 代码部分&#xff1a; from PIL import Image import matplotlib.pypl…

CSS实现实现当文本内容过长时,中间显示省略号...,两端正常展示

HTML 结构解析 文档结构: <ul class"con">: 一个无序列表&#xff0c;包含多个列表项。 每个 <li class"wrap"> 表示一个列表项&#xff0c;内部有两个 <span> 元素&#xff1a; <span class"txt">: 显示文本内容。<…

ROS VRRP软路由双线组网方式

虚拟路由冗余协议 Virtual Router Redundancy Protocol (VRRP)&#xff0c;MikroTik RouteROS VRRP 协议遵循 RFC 2338。 VRRP 协议是保证访问一些资源不会中断&#xff0c;即通过多台路由器组成一个网关集合&#xff0c;如果其中一台路由器出现故障&#xff0c;会自动启用另外…

设计编程网站集:简述可扩展性系统设计(笔记)

视频连接&#xff1a;简述可扩展性系统设计 三个关键原则 无状态 松散耦合 异步处理 扩展 负载均衡 缓存 分片

openCV与eigen两种方法---旋转向量转旋转矩阵

#include <Eigen/Dense> #include <opencv2/core/eigen.hpp> #include <opencv2/opencv.hpp> using namespace cv; using namespace std; int main() {// opencv 旋转向量cv::Vec3d rvec(1.0, 2.0, 3.0);cv::Mat rotation_matrix;cv::Rodrigues(rvec, rotati…

卷积运算和卷积定理

卷积运算 卷积运算是信号处理、图像处理和深度学习中的核心概念&#xff0c;用于表示两个函数之间的相互作用。它将一个函数通过滑动窗口的方式与另一个函数结合&#xff0c;产生一个新的函数&#xff0c;反映两者的重叠程度。 1. 定义 连续信号的卷积&#xff1a; 给定两个连…

【板间连接器焊接】

一、背景 近期工作需要,用到了AX7Z020核心板(黑金),官网链接:https://www.alinx.com/detail/271。 板子打好之后,遇到了焊接问题。对自身焊接技术还是比较自信的,直接上去焊接了2个连接器。拖锡搞了3小时后,放弃了。热风枪1分钟不到就把连接器吹下来了,看引脚90%都是…

低代码开发平台搭建思考与实战

什么是低代码开发平台&#xff1f; 低代码开发平台是一种平台软件&#xff0c;人们能通过它提供的图形化配置功能&#xff0c;快速配置出满足各种特定业务需求的功能软件。 具有以下特点&#xff1a; 提供可视化界面进行程序开发0代码或少量代码快速生成应用 什么是低代码产…

React Native 基础

React 的核心概念 定义函数式组件 import组件 要定义一个Cat组件,第一步要使用 import 语句来引入React以及React Native的 Text 组件: import React from react; import { Text } from react-native; 定义函数作为组件 const CatApp = () => {}; 渲染Text组件

ftdi_sio应用学习笔记 3 - GPIO

目录 1. 查找gpiochip 2. 打开GPIO 2.1 libgpiod库方式 2.2 系统方式 3. 关闭GPIO 3.1 libgpiod库方式 3.2 系统方式 4. 设置方向 4.1 libgpiod库方式 4.2 系统方式 5. 设置GPIO电平 5.1 libgpiod库方式 5.2 系统方式 6. 读取GPIO电平 6.1 libgpiod库方式 6.2 …

微信小程序登录注册页面设计(小程序项目)

需求 在微信小程序设计并实现登录页面&#xff0c;并填写相关登录注册函数 实现效果 代码实现 html代码 <view class"top" style"border-bottom-style: none;background-color:#FF8C69;"><!-- <view class"back" bind:tap"…

神经网络(系统性学习三):多层感知机(MLP)

相关文章&#xff1a; 神经网络中常用的激活函数 神经网络&#xff08;系统性学习一&#xff09;&#xff1a;入门篇 神经网络&#xff08;系统性学习二&#xff09;&#xff1a;单层神经网络&#xff08;感知机&#xff09; 多层感知机&#xff08;MLP&#xff09; 多层感…

Android 14 screenrecord录制视频失败的原因分析

文章目录 1. 权限问题2. 存储空间不足3. 命令被中断4. 目标路径问题5. Android 14 的新限制6. 文件系统同步问题7. 录制失败检查步骤总结&#xff1a; 在 Android 14 系统上&#xff0c;使用 screenrecord 命令录制视频后&#xff0c;生成的文件大小为 0&#xff0c;可能的原因…

Uniapp 简单配置鸿蒙

Uniapp 简单配置鸿蒙 前言下载并配置鸿蒙IDEHbuilder X 配置基本的信息生成相关证书登录官网获取证书IDE配置证书添加调试设备可能出现的问题前言 如今鸿蒙的盛起,作为多端开发的代表也是开始兼容鸿蒙应用的开发,接下来我将介绍如何在uniapp中配置鸿蒙。 注意:hbuilder X的…

git使用(一)

git使用&#xff08;一&#xff09; 为什么学习git?两种版本控制系统在github上创建一个仓库&#xff08;repository&#xff09;windows上配置git环境在Linux上配置git环境 为什么学习git? 代码写了好久不小心删了&#xff0c;可以使用git防止&#xff0c;每写一部分代码通…

C# 数据结构之【树】C#树

以二叉树为例进行演示。二叉树每个节点最多有两个子节点。 1. 新建二叉树节点模型 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace DataStructure {class TreeNode{public int Data { get;…

HarmonyOs鸿蒙开发实战(20)=>一文学会基础使用组件导航Navigation

敲黑板&#xff0c;以下是重点技巧。文章末尾有实战项目效果截图及代码截图可参考 1.概要 Navigation是路由导航的根视图容器Navigation组件主要包含​导航页&#xff08;NavBar&#xff09;和子页&#xff08;NavDestination&#xff09;&#xff0c;导航页不存在页面栈中&am…

python从入门到精通:pyspark实战分析

前言 spark&#xff1a;Apache Spark是用于大规模数据&#xff08;large-scala data&#xff09;处理的统一&#xff08;unified&#xff09;分析引擎。简单来说&#xff0c;Spark是一款分布式的计算框架&#xff0c;用于调度成本上千的服务器集群&#xff0c;计算TB、PB乃至E…