Map的keySet()方法和entrySet()方法(java学习)

文章目录

前言

Map.keySet() 是 Java 集合框架中非常基础、但也极其容易踩坑的一个方法。

简单来说,它的直接作用是返回 Map 中所有键(Key)组成的一个 Set 集合

因为 Map 的键是不可重复的,所以返回值是一个 Set 类型在逻辑上非常严密。

但要真正掌握这个方法,核心在于理解计算机科学中的一个重要设计模式:视图(View)

一、视图核心概念

它是"视图",而不是"副本"

调用keySet() 并没有在内存中开辟一块新空间去复制原有的 Key 从而生成一个新的 Set。

它返回的实际上是原 Map 内部结构的一个"透视镜"或"影子"。

这意味着,返回的 Set 和原 Map 是强绑定的

所以 keySet()返回的Set集合只能用于遍历,不能做修改,否则会破坏原来Map数据

二、视图带来的三大联动法则

由于这种底层绑定关系,对原 Map 或返回的 Set 进行操作时,必须遵循以下规则:

  • Map 变,Set 跟着变: 如果你往原 Map 里 put 了一个新键值对,或者 remove 了一个键值对,你去遍历那个 Set 时,会发现里面的元素已经自动同步更新了。
  • 删 Set 里的元素,Map 里的键值对也会消失: 如果你调用 Set 的 remove()removeAll()clear() 方法,或者通过迭代器(Iterator)的 remove() 删除了某个 Key,原 Map 里对应的那个 Key 及其关联的 Value 会被一并强制删除。
  • 绝对禁止往 Set 里添加元素(核心考点): 如果你尝试调用 Set 的 add()addAll() 方法,程序在运行时会直接抛出 UnsupportedOperationException 异常。
    • 底层逻辑: Map 存储的是完整的键值对(Key-Value)。如果你只往 keySet 暴露的 Set 里强行塞入一个 Key,Java 根本不知道该给这个孤立的 Key 分配什么 Value,因此底层直接屏蔽了添加操作。

三、基本使用场景

通常我们用它来遍历 Map 中的键,或者需要将所有的键作为一个集合传递给其他方法:

Java 复制代码
Map<String, Integer> map = new HashMap<>();
map.put("Alice", 25);
map.put("Bob", 30);

// 获取所有键的视图集合
Set<String> keys = map.keySet();

// 仅遍历键
for (String key : keys) {
    System.out.println("Key: " + key);
}

四、性能考量(进阶)

在工程实践中,如果你遍历 Map 仅仅是为了使用 Key ,那么使用 keySet() 是最简洁优雅的做法。

但如果在遍历的过程中既需要 Key 又需要 Value,就要格外小心了。

❌ 容易引发性能损耗的写法:

Java 复制代码
for (String key : map.keySet()) {
    // 每次都要重新对 key 进行 hash 计算并寻址找 value
    Integer value = map.get(key); 
    System.out.println("Key: " + key + ", Value: " + value);
}

虽然哈希表通过 Key 查找 Value 的理论时间复杂度是 O ( 1 ) O(1) O(1),但每次 map.get(key) 依然会在底层执行哈希计算、数组寻址、甚至是红黑树/链表的遍历。在数据量庞大时,这是一种非常多余的消耗。

✅ 高效的规范写法:

当需要同时操作键和值时,应该使用 entrySet() 替代 keySet()

Java 复制代码
for (Map.Entry<String, Integer> entry : map.entrySet()) {
    // 直接从节点对象中获取 key 和 value,没有额外的查找开销
    System.out.println("Key: " + entry.getKey() + ", Value: " + entry.getValue());
}

为什么快?

底层遍历 Map 的数据节点(Node)时,顺手就把封装在一起的 Key 和 Value 都拿到了,完全省去了 get() 方法的二次哈希寻址过程。这是编写高质量 Java 代码的重要细节。

如果使用的是 Java 8 及以上版本,推荐使用更现代的 forEach 语法(底层原理也是基于 entrySet),代码更加简洁:

java 复制代码
// 最优雅的双值遍历方式
map.forEach((key, value) -> {
    System.out.println("Key: " + key + ", Value: " + value);
});

总结

  1. Map的keySet()方法返回Map中所有键(key)的集合,且每一个元素不重复;
  2. keySet()entrySet() 返回的都是视图(View) ,对视图的删除操作会影响原 Map,但严禁执行添加操作
  3. 当你只需要 Key 时,使用 keySet()
  4. 出于性能考虑,当你需要同时获取 Key 和 Value 时,坚决使用 entrySet() 替代 keySet() + get()
  5. 拥抱新特性,日常开发中强推 map.forEach((k, v) -> {}),不仅性能好,而且代码可读性极佳。
相关推荐
探险的机器猫1 小时前
使用 java 搭建一个基于 StreamableHTTP 的 MCP 服务
java·mcp
乐之者v1 小时前
AI编程-- codex并行开发需求
java
weixin_520649871 小时前
C#线程底层原理知识
java·开发语言
xieliyu.2 小时前
Java手搓数据结构:从零模拟实现单向无头非循环链表
java·数据结构·学习·链表
0xDevNull2 小时前
队列(Queue)实战教程:从原理到架构应用
java·开发语言·后端
再写一行代码就下班2 小时前
word模版导出(占位符方式)
java·开发语言·word
懒得起名_yyf2 小时前
AI智能体Skills全面入门指南
java
敖正炀2 小时前
集合-List-ArrayList
java