JAVA 集合框架进阶:Map 接口的深度解析与实战

JAVA 集合框架进阶:Map 接口的深度解析与实战

JAVA 集合框架进阶:Map 接口的深度解析与实战

在这里插入图片描述

1.1 本章学习目标与重点

💡 掌握 Map 接口的核心特性,理解 Key-Value 键值对的存储结构与设计思想。
💡 熟练掌握 HashMap、LinkedHashMap、TreeMap 等实现类的底层原理与适用场景。
💡 理解 Map 集合的线程安全问题,掌握并发环境下的解决方案。
⚠️ 本章重点是 HashMap 的底层实现原理不同 Map 实现类的性能对比,这是面试和开发中的高频核心考点。

1.2 Map 接口核心概述

1.2.1 Map 接口的定义与特性

💡 Map 是一种键值对(Key-Value) 集合,它的核心是通过键(Key)来唯一标识值(Value)。
Map 接口中的 Key 具有唯一性,不能重复;Value 可以重复,并且可以为 null
Map 接口与 Collection 接口是并列关系,它不属于 Collection 体系,没有继承关系。

Map 接口的核心方法:

  • put(K key, V value):添加键值对,Key 重复时会覆盖原有 Value
  • get(Object key):根据 Key 获取 Value,Key 不存在时返回 null
  • remove(Object key):根据 Key 删除对应的键值对
  • containsKey(Object key):判断是否包含指定 Key
  • containsValue(Object value):判断是否包含指定 Value
  • keySet():获取所有 Key 组成的 Set 集合
  • values():获取所有 Value 组成的 Collection 集合
  • entrySet():获取所有键值对(Map.Entry)组成的 Set 集合

✅ 核心结论:Map 适合通过唯一标识(Key)快速查找对应数据(Value)的场景。

1.2.2 Map 集合的遍历方式

Map 集合有三种常用的遍历方式,我们以 HashMap 为例进行代码实操:

importjava.util.HashMap;importjava.util.Map;importjava.util.Set;publicclassMapTraversalDemo{publicstaticvoidmain(String[] args){Map<String,String> map =newHashMap<>(); map.put("name","张三"); map.put("age","20"); map.put("gender","男");// 方式1:遍历所有Key,通过Key获取ValueSystem.out.println("方式1:遍历Key获取Value");Set<String> keySet = map.keySet();for(String key : keySet){String value = map.get(key);System.out.println(key +"="+ value);}// 方式2:遍历所有键值对(Map.Entry)System.out.println("\n方式2:遍历Map.Entry");Set<Map.Entry<String,String>> entrySet = map.entrySet();for(Map.Entry<String,String> entry : entrySet){System.out.println(entry.getKey()+"="+ entry.getValue());}// 方式3:JDK8+ Lambda表达式遍历System.out.println("\n方式3:Lambda表达式遍历"); map.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

方式1:遍历Key获取Value name=张三 age=20 gender=男 方式2:遍历Map.Entry name=张三 age=20 gender=男 方式3:Lambda表达式遍历 name=张三 age=20 gender=男 

✅ 核心结论:遍历大量数据时,方式2(entrySet)效率最高,因为它避免了通过 Key 重复查询 Value 的操作。

1.3 HashMap:基于哈希表的实现

1.3.1 HashMap 底层原理(JDK8)

💡 JDK8 中 HashMap 的底层结构是 数组 + 链表 + 红黑树 的组合结构,目的是解决哈希冲突,提升查询效率。

  1. 数组(哈希桶):数组的每个元素是一个链表或红黑树的头节点,默认初始容量为 16,默认加载因子为 0.75。
  2. 链表:当多个 Key 的哈希值相同,且对应数组下标位置已有元素时,会以链表形式存储,称为哈希冲突
  3. 红黑树:当链表长度超过阈值(默认为 8),并且数组长度大于等于 64 时,链表会转换为红黑树,将查询时间复杂度从 O(n) 降低到 O(log n)

HashMap 的核心存储流程:
① 📝 计算 Key 的哈希值:通过 hash(key) 方法计算,目的是降低哈希冲突概率。
② 📝 计算数组下标:(数组长度 - 1) & 哈希值,等价于取模运算但效率更高。
③ 📝 判断下标位置是否为空:为空则直接插入新节点;不为空则判断 Key 是否重复。
④ 📝 处理 Key 重复:Key 重复则覆盖 Value;不重复则插入链表或红黑树。
⑤ 📝 扩容判断:当元素个数超过 数组容量 * 加载因子 时,数组会扩容为原来的 2 倍。

1.3.2 代码实操:HashMap 的常用操作

importjava.util.HashMap;importjava.util.Map;publicclassHashMapDemo{publicstaticvoidmain(String[] args){Map<String,Integer> hashMap =newHashMap<>();// 1. 添加键值对 hashMap.put("语文",90); hashMap.put("数学",95); hashMap.put("英语",92); hashMap.put("数学",100);// Key重复,覆盖原有ValueSystem.out.println("HashMap内容:"+ hashMap);// 2. 根据Key获取ValueInteger mathScore = hashMap.get("数学");System.out.println("数学成绩:"+ mathScore);// 3. 判断是否包含指定Key或Valueboolean hasEnglish = hashMap.containsKey("英语");boolean has90 = hashMap.containsValue(90);System.out.println("包含英语Key:"+ hasEnglish);System.out.println("包含90分Value:"+ has90);// 4. 删除键值对 hashMap.remove("语文");System.out.println("删除语文后的HashMap:"+ hashMap);// 5. 获取集合大小int size = hashMap.size();System.out.println("HashMap大小:"+ size);// 6. 清空集合 hashMap.clear();System.out.println("清空后是否为空:"+ hashMap.isEmpty());}}

输出结果

HashMap内容:{语文=90, 数学=100, 英语=92} 数学成绩:100 包含英语Key:true 包含90分Value:true 删除语文后的HashMap:{数学=100, 英语=92} HashMap大小:2 清空后是否为空:true 

1.3.3 自定义对象作为 Key 的注意事项

⚠️ 当使用自定义对象作为 HashMap 的 Key 时,必须重写 hashCode()equals() 方法,否则无法保证 Key 的唯一性。

我们以 Student 类为例,实现基于学号的 Key 唯一性:

importjava.util.HashMap;importjava.util.Map;importjava.util.Objects;classStudent{privateString id;privateString name;publicStudent(String id,String name){this.id = id;this.name = name;}// 重写equals方法:根据学号判断Key是否相同@Overridepublicbooleanequals(Object o){if(this== o)returntrue;if(o ==null||getClass()!= o.getClass())returnfalse;Student student =(Student) o;returnObjects.equals(id, student.id);}// 重写hashCode方法:根据学号计算哈希值@OverridepublicinthashCode(){returnObjects.hash(id);}@OverridepublicStringtoString(){return"Student{id='"+ id +"',+ name +"'}";}}publicclassHashMapCustomKeyDemo{publicstaticvoidmain(String[] args){Map<Student,String> studentMap =newHashMap<>();Student s1 =newStudent("001","张三");Student s2 =newStudent("002","李四");Student s3 =newStudent("001","张三");// 与s1学号相同 studentMap.put(s1,"一班"); studentMap.put(s2,"二班"); studentMap.put(s3,"三班");// Key重复,覆盖s1的Value// 遍历输出 studentMap.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

Student{id='001', name='张三'}=三班 Student{id='002', name='李四'}=二班 

1.3.4 HashMap 性能分析

  • 增删查操作:理想情况下时间复杂度为 O(1),哈希冲突严重时会退化为 O(n),红黑树转换后为 O(log n)
  • 扩容机制:扩容时需要重新计算所有元素的下标,非常消耗性能,开发中建议提前指定初始容量,减少扩容次数。
    ⚠️ 注意事项:HashMap 是线程不安全的集合,多线程环境下使用会出现数据错乱或 ConcurrentModificationException 异常。

1.4 LinkedHashMap:有序的哈希表

1.4.1 LinkedHashMap 底层原理

💡 LinkedHashMap 是 HashMap 的子类,底层结构是 HashMap + 双向链表
它通过双向链表维护键值对的插入顺序访问顺序,保证遍历顺序与插入顺序一致,或者与最近访问顺序一致。
LinkedHashMap 的元素唯一性判断规则与 HashMap 完全相同。

1.4.2 代码实操1:插入顺序模式(默认)

importjava.util.LinkedHashMap;importjava.util.Map;publicclassLinkedHashMapInsertOrderDemo{publicstaticvoidmain(String[] args){Map<String,String> linkedHashMap =newLinkedHashMap<>(); linkedHashMap.put("b","B"); linkedHashMap.put("a","A"); linkedHashMap.put("c","C");// 遍历顺序与插入顺序一致 linkedHashMap.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

b=B a=A c=C 

1.4.3 代码实操2:访问顺序模式

通过构造方法 LinkedHashMap(int initialCapacity, float loadFactor, boolean accessOrder) 可以开启访问顺序模式,最近访问的元素会被移到链表尾部。

importjava.util.LinkedHashMap;importjava.util.Map;publicclassLinkedHashMapAccessOrderDemo{publicstaticvoidmain(String[] args){// 开启访问顺序模式:accessOrder = trueMap<String,String> linkedHashMap =newLinkedHashMap<>(16,0.75f,true); linkedHashMap.put("a","A"); linkedHashMap.put("b","B"); linkedHashMap.put("c","C");System.out.println("初始顺序:"); linkedHashMap.forEach((key, value)->System.out.println(key +"="+ value));// 访问元素a,触发访问顺序调整 linkedHashMap.get("a");System.out.println("\n访问元素a后的顺序:"); linkedHashMap.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

初始顺序: a=A b=B c=C 访问元素a后的顺序: b=B c=C a=A 

✅ 核心结论:访问顺序模式的 LinkedHashMap 可以用来实现 LRU 缓存淘汰算法(最近最少使用淘汰)。

1.4.4 性能分析

  • LinkedHashMap 的增删查效率略低于 HashMap,因为需要维护双向链表的节点引用。
  • 适合需要有序遍历高效查找的场景,例如缓存系统、配置参数存储等。
    ⚠️ 注意事项:LinkedHashMap 同样是线程不安全的集合。

1.5 TreeMap:基于红黑树的排序映射

1.5.1 TreeMap 底层原理

💡 TreeMap 的底层结构是红黑树,它会自动对 Key 进行排序,默认是升序排列。
TreeMap 不允许 Key 为 null,因为排序时会抛出空指针异常。
TreeMap 保证 Key 唯一性的方式是通过比较 Key 的大小,而不是 hashCode()equals() 方法。

TreeMap 的两种排序方式:

  1. 自然排序:Key 实现 Comparable 接口,重写 compareTo() 方法。
  2. 定制排序:创建 TreeMap 时传入 Comparator 比较器,自定义排序规则。

1.5.2 代码实操1:自然排序

importjava.util.Map;importjava.util.TreeMap;publicclassTreeMapNaturalSortDemo{publicstaticvoidmain(String[] args){Map<Integer,String> treeMap =newTreeMap<>(); treeMap.put(3,"C"); treeMap.put(1,"A"); treeMap.put(2,"B");// 自动按Key升序排列 treeMap.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

1=A 2=B 3=C 

1.5.3 代码实操2:定制排序

我们对字符串 Key 进行降序排列,通过 Comparator 实现定制排序:

importjava.util.Comparator;importjava.util.Map;importjava.util.TreeMap;publicclassTreeMapCustomSortDemo{publicstaticvoidmain(String[] args){// 传入比较器,实现Key降序排列Map<String,Integer> treeMap =newTreeMap<>(newComparator<String>(){@Overridepublicintcompare(String o1,String o2){return o2.compareTo(o1);// 降序排列}}); treeMap.put("Java",10); treeMap.put("Python",8); treeMap.put("Go",9); treeMap.forEach((key, value)->System.out.println(key +"="+ value));}}

输出结果

Python=8 Java=10 Go=9 

1.5.4 性能分析

  • 增删查操作:时间复杂度稳定为 O(log n),效率低于 HashMap,但支持有序遍历。
  • 适合需要排序范围查询的场景,例如排行榜、字典排序等。
    ⚠️ 注意事项:
  1. TreeMap 是线程不安全的集合。
  2. 存储自定义对象作为 Key 时,必须指定排序规则,否则会抛出 ClassCastException

1.6 Hashtable:线程安全的哈希表

1.6.1 Hashtable 核心特性

💡 Hashtable 是 Map 接口的早期实现类,底层结构是 数组 + 链表(JDK8 没有红黑树优化)。
它的所有方法都添加了 synchronized 关键字,是线程安全的集合。
Hashtable 不允许 Key 或 Value 为 null,默认初始容量为 11,加载因子为 0.75。

1.6.2 性能分析

  • Hashtable 的线程安全是通过方法加锁实现的,锁粒度大,并发性能低。
  • 现代开发中,不推荐使用 Hashtable,优先使用 ConcurrentHashMap 实现线程安全。

1.7 ConcurrentHashMap:并发安全的哈希表

1.7.1 ConcurrentHashMap 核心特性

💡 ConcurrentHashMap 是 JUC 包下的线程安全集合,专门用于解决 HashMap 的并发问题。
JDK8 中 ConcurrentHashMap 的底层结构是 数组 + 链表 + 红黑树,与 HashMap 类似。
它采用分段锁CAS + synchronized 的方式实现线程安全,锁粒度小,并发性能远高于 Hashtable。
ConcurrentHashMap 支持 Key 和 Value 为 null(与 Hashtable 不同)。

1.7.2 代码实操:ConcurrentHashMap 的使用

importjava.util.Map;importjava.util.concurrent.ConcurrentHashMap;publicclassConcurrentHashMapDemo{publicstaticvoidmain(String[] args){Map<String,Integer> concurrentMap =newConcurrentHashMap<>();// 多线程环境下安全操作newThread(()->{for(int i =0; i <1000; i++){ concurrentMap.put("thread1_"+ i, i);}}).start();newThread(()->{for(int i =0; i <1000; i++){ concurrentMap.put("thread2_"+ i, i);}}).start();// 等待线程执行完成try{Thread.sleep(1000);}catch(InterruptedException e){ e.printStackTrace();}System.out.println("ConcurrentHashMap大小:"+ concurrentMap.size());}}

输出结果

ConcurrentHashMap大小:2000 

✅ 核心结论:多线程环境下优先使用 ConcurrentHashMap,兼顾线程安全和并发性能。

1.8 实战案例:基于 Map 实现 LRU 缓存

1.8.1 需求分析

💡 实现一个 LRU(最近最少使用)缓存工具类,满足以下需求:

  1. 缓存容量有限,超出容量时自动淘汰最近最少使用的元素。
  2. 支持缓存的添加、查询、删除操作。
  3. 保证操作的时间复杂度尽可能低。

1.8.2 实现思路

利用 LinkedHashMap 的访问顺序模式实现 LRU 缓存,重写 removeEldestEntry() 方法,自定义淘汰规则。

1.8.3 代码实现

importjava.util.LinkedHashMap;importjava.util.Map;/** * 基于LinkedHashMap实现的LRU缓存 */publicclassLRUCache<K,V>extendsLinkedHashMap<K,V>{// 缓存最大容量privatefinalint maxCapacity;// 构造方法:开启访问顺序模式publicLRUCache(int maxCapacity){super(16,0.75f,true);this.maxCapacity = maxCapacity;}/** * 重写该方法,自定义淘汰规则 * @param eldest 最久未使用的元素 * @return true表示淘汰该元素,false表示不淘汰 */@OverrideprotectedbooleanremoveEldestEntry(Map.Entry<K,V> eldest){// 当元素个数超过最大容量时,淘汰最久未使用的元素returnsize()> maxCapacity;}// 测试方法publicstaticvoidmain(String[] args){LRUCache<String,Integer> cache =newLRUCache<>(3);// 添加缓存元素 cache.put("A",1); cache.put("B",2); cache.put("C",3);System.out.println("初始缓存:"+ cache);// 访问元素A,调整访问顺序 cache.get("A");System.out.println("访问A后的缓存:"+ cache);// 添加元素D,超出容量,淘汰最久未使用的B cache.put("D",4);System.out.println("添加D后的缓存:"+ cache);}}

输出结果

初始缓存:{A=1, B=2, C=3} 访问A后的缓存:{B=2, C=3, A=1} 添加D后的缓存:{C=3, A=1, D=4} 

1.8.4 案例总结

✅ 这个 LRU 缓存工具类充分利用了 LinkedHashMap 的特性,代码简洁且性能高效。
通过重写 removeEldestEntry() 方法,轻松实现了缓存淘汰规则,在实际开发中可直接用于本地缓存场景。

1.9 本章总结

  1. Map 是键值对集合,Key 唯一,Value 可重复,常用实现类有 HashMap、LinkedHashMap、TreeMap。
  2. HashMap 底层是数组+链表+红黑树,查询效率高,适合快速查找场景,线程不安全。
  3. LinkedHashMap 基于 HashMap+双向链表,支持插入顺序或访问顺序遍历,可实现 LRU 缓存。
  4. TreeMap 底层是红黑树,支持 Key 排序,适合有序遍历和范围查询场景,线程不安全。
  5. 多线程环境下,优先使用 ConcurrentHashMap 保证线程安全,避免使用 Hashtable。
  6. 自定义对象作为 HashMap Key 时,必须重写 hashCode()equals() 方法。

Read more

《C++ 递归、搜索与回溯》第2-3题:合并两个有序链表,反转链表

《C++ 递归、搜索与回溯》第2-3题:合并两个有序链表,反转链表

🔥个人主页:Cx330🌸 ❄️个人专栏:《C语言》《LeetCode刷题集》《数据结构-初阶》《C++知识分享》 《优选算法指南-必刷经典100题》《Linux操作系统》:从入门到入魔 《Git深度解析》:版本管理实战全解 🌟心向往之行必能至 🎥Cx330🌸的简介: 目录 前言: 2. 合并两个有序链表 算法原理(递归): 思路: 解法代码(C++): 博主手记(字体还请见谅哈): 3. 反转链表 算法原理(递归): 思路: 解法代码(C++): 博主手记(字体还请见谅哈): 结尾: 前言: 聚焦算法题实战,系统讲解三大核心板块:“精准定位最优解”——优选算法,“简化逻辑表达,系统性探索与剪枝优化”——递归与回溯,“以局部最优换全局高效”——贪心算法,讲解思路与代码实现,帮助大家快速提升代码能力 2.

By Ne0inhk
【C++初阶】C++入门相关知识(2):输入输出 & 缺省参数 & 函数重载

【C++初阶】C++入门相关知识(2):输入输出 & 缺省参数 & 函数重载

🎈主页传送门:良木生香 🔥个人专栏:《C语言》 《数据结构-初阶》 《程序设计》《鼠鼠的C++学习之路》 🌟人为善,福随未至,祸已远行;人为恶,祸虽未至,福已远离 上期回顾:在上一篇文章中,我们对C++进行了初步的认识,学习了C++的发展历史,第一个C++程序以及命名空间,我们知道,C++的出现就是为了改进和完善C语言的不足,使得程序更加高效,程序员编写起来更加方便快捷,那么本篇文章我们继续往下认识C++的入门相关知识 目录 一、C++的输入&输出 1.1、核心载体:头文件 1.2、核心的IO对象:cin与cout 1.2.1、std::cin 标准输入流 1.

By Ne0inhk
C++ 多线程同步之条件变量(condition_variable)实战

C++ 多线程同步之条件变量(condition_variable)实战

C++ 多线程同步之条件变量(condition_variable)实战 💡 学习目标:掌握 C++ 标准库中条件变量的使用方法,理解条件变量与互斥锁的协同工作机制,能够解决多线程间的等待-通知问题。 💡 学习重点:std::condition_variable 的核心接口、wait() 与 notify_one()/notify_all() 的配合使用、生产者-消费者模型的实现。 49.1 条件变量的引入场景 在多线程编程中,我们经常会遇到线程需要等待某个条件满足后再执行的场景。 比如生产者线程生产数据后,消费者线程才能消费;队列不为空时,消费者才能从中取数据。 如果仅用互斥锁实现,消费者线程只能不断轮询检查条件,这会造成 CPU 资源的浪费。 ⚠️ 注意事项:单纯的轮询会导致 CPU 空转,降低程序运行效率,条件变量就是为解决这类问题而生的。 举个简单的轮询反例,消费者不断检查队列是否有数据: #include<iostream>

By Ne0inhk
JSP技术入门指南【一】利用IDEA从零开始搭建你的第一个JSP系统

JSP技术入门指南【一】利用IDEA从零开始搭建你的第一个JSP系统

Jsp技术入门指南【一】利用IDEA从零开始搭建你的第一个JSP系统 * 前言 * 一、什么是JSP * 1.1 JSP是干什么的? * 1.2 JSP与Servlet的关系是什么? * 二、在Idea中创建第一个JSP系统 * 三、JSP和HTML的差别 * 3.1 格式区别 * 3.2 注释区别 前言 * 在前面的内容中,我们已经系统学习了 Web 开发的基础技术:从构建网页骨架的 HTML、美化页面的 CSS,到实现服务器端逻辑的 Java Servlet。 * 这些知识为我们打开了动态 Web 开发的大门,让我们能够通过 Servlet 处理客户端请求、操作数据库并返回动态数据。 * 然而,在 Servlet 中直接拼接 HTML 代码实现页面渲染时,代码往往显得繁琐且难以维护 —— 有没有一种更简洁、更直观的方式,

By Ne0inhk