Java中HashMap的使用操作

 更新时间:2025年08月26日 09:38:54   作者:熙客  
HashMap位于java.util包下,它实现了Map接口,提供了基于键值对(Key-Value)的数据存储方式,并允许使用null作为键和值,这篇文章主要介绍了Java中HashMap的使用,需要的朋友可以参考下

一、概念

HashMap 位于 java.util 包下。它实现了 Map 接口,提供了基于键值对(Key-Value)的数据存储方式,并允许使用 null 作为键和值。

核心思想:使用“键(Key)”的哈希值来存储和检索“值(Value)”。

核心特点:

  • 基于哈希表:它使用哈希算法来存储和检索数据,这使得在理想情况下,get 和 put 操作的时间复杂度可以达到 O(1)
  • 无序HashMap 不保证其中元素的顺序(即插入顺序和访问顺序),并且顺序也可能会随时间(如扩容时)而变化。如果需要有序,可以使用 LinkedHashMap
  • 非线程安全:多个线程同时操作一个 HashMap 可能会导致数据不一致。如果多个线程同时访问一个 HashMap 并至少有一个线程修改了它,则必须在外部进行同步。或者使用 Collections.synchronizedMap() 进行包装,或者更推荐使用 ConcurrentHashMap

二、常用操作

2.1 初始化

// 1. 最常见的无参构造,默认初始容量16,负载因子0.75
HashMap<String, Integer> map = new HashMap<>();
// 2. 指定初始容量(减少扩容次数,优化性能)
HashMap<String, Integer> mapWithCapacity = new HashMap<>(32);
// 3. 指定初始容量和负载因子(高级用法,通常不需要)
HashMap<String, Integer> mapWithFactor = new HashMap<>(32, 0.8f);
// 4. 通过另一个Map来创建
HashMap<String, Integer> anotherMap = new HashMap<>(map);

扩容机制:扩充为原数组容量的2倍

当 HashMap 中的元素数量size超过当前阈值(threshold) 时,就会触发扩容。

阈值 (threshold) = 容量 (capacity) * 负载因子 (load factor)

  • 默认示例:默认容量为 16,默认负载因子为 0.75。
    • 那么阈值就是 16 * 0.75 = 12
    • 当执行 put() 操作后,size 变得大于 12(即 13)时,就会触发扩容。

2.2 CRUD

HashMap<String, String> capitalCities = new HashMap<>();
// 添加与更新
// put(K key, V value) - 添加键值对,如果key已存在,则更新其value
capitalCities.put("USA", "Washington D.C.");
capitalCities.put("Germany", "Berlin");
capitalCities.put("Germany", "Berlin"); // 重复放入,不会改变
capitalCities.put("Germany", "New Berlin"); // Key已存在,Value会被更新为 "New Berlin"
// putIfAbsent(K key, V value) - (Java 8+) 只有在key不存在或对应的value为null时,才放入
capitalCities.putIfAbsent("France", "Paris"); // 会放入,因为France不存在
capitalCities.putIfAbsent("Germany", "Paris"); // 不会放入,因为Germany已存在,Value仍然是"New Berlin"
// 获取元素
// get(Object key) - 根据key获取value,如果key不存在,返回null
String capitalOfGermany = capitalCities.get("Germany"); // "New Berlin"
String capitalOfJapan = capitalCities.get("Japan"); // null
// getOrDefault(Object key, V defaultValue) - (Java 8+) key不存在时返回一个默认值
String capitalOfJapanSafe = capitalCities.getOrDefault("Japan", "Not Found"); // "Not Found"
// 检查元素是否存在
// containsKey(Object key) - 检查某个key是否存在
boolean hasGermany = capitalCities.containsKey("Germany"); // true
boolean hasJapan = capitalCities.containsKey("Japan"); // false
// containsValue(Object value) - 检查某个value是否存在(效率较低,需要遍历)
boolean hasParis = capitalCities.containsValue("Paris"); // true
// 删除元素
// remove(Object key) - 根据key删除键值对,返回被删除的value
String removedValue = capitalCities.remove("Germany"); // removedValue = "New Berlin"
// remove(Object key, Object value) - (Java 8+) 只有当key和value都匹配时才删除
boolean isRemoved = capitalCities.remove("USA", "LA"); // false, 因为Value不匹配"Washington D.C.",删除失败
boolean isRemoved2 = capitalCities.remove("USA", "Washington D.C."); // true, 删除成功

2.3 遍历

// 遍历所有键:keySet()
for (String country : capitalCities.keySet()) {
    System.out.println("Country: " + country);
    // 可以通过key再get value,但效率较低(不推荐在循环内这样用)
    // System.out.println("Capital: " + capitalCities.get(country));
}
// 遍历所有值:values()
for (String capital : capitalCities.values()) {
    System.out.println("Capital: " + capital);
}
// 遍历所有键值对:entrySet() (最推荐、最高效的方式)
for (Map.Entry<String, String> entry : capitalCities.entrySet()) {
    String country = entry.getKey();
    String capital = entry.getValue();
    System.out.println(country + " -> " + capital);
}
// 使用 Java 8 forEach + Lambda 表达式 (最简洁)
capitalCities.forEach((country, capital) -> {
    System.out.println(country + " -> " + capital);
});

2.4 其他常用方法

// size() - 返回键值对的数量
int size = capitalCities.size();
// isEmpty() - 判断是否为空
boolean isEmpty = capitalCities.isEmpty();
// clear() - 清空所有映射
capitalCities.clear();
// replace(K key, V oldValue, V newValue) - (Java 8+) 替换操作
capitalCities.replace("France", "Paris", "Lyon"); // 只有当旧值匹配时才替换

三、与其他Map的比较

特性HashMapLinkedHashMapTreeMapHashtableConcurrentHashMap
排序保证无顺序插入顺序 或 访问顺序 (LRU)键的自然顺序 或 自定义比较器顺序无顺序无顺序
是否允许 null允许 一个 null key 和多个 null value允许 一个 null key 和多个 null value不允许 null key (取决于Comparator)不允许 null key 或 null value不允许 null key 或 null value
线程安全 (非同步) (非同步) (非同步) (同步,每个方法都用 synchronized 修饰) (采用分段锁/CAS等更高效的并发控制)
性能特点O(1) 时间复杂度的 get/put (平均情况)比 HashMap 稍慢,因为要维护链表O(log n) 时间复杂度的 get/put类似 HashMap,但同步开销大,性能差高并发性能极佳,读操作通常无需锁
底层实现数组 + 链表/红黑树 (哈希表)HashMap + 双向链表 (维护顺序)红黑树数组 + 链表 (哈希表)数组 + 链表/红黑树 + CAS + 分段锁 (JDK 7/8 不同)
迭代顺序一致性不保证,甚至可能随时间变化保证,与插入顺序或访问顺序一致保证,根据键的顺序排序不保证不保证
引入版本JDK 1.2JDK 1.4JDK 1.2JDK 1.0 (是古老遗留类)JDK 1.5 (java.util.concurrent 包)

到此这篇关于Java中HashMap的使用操作的文章就介绍到这了,更多相关Java HashMap内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java Comparator比较器实例解析

    Java Comparator比较器实例解析

    这篇文章主要介绍了Java Comparator比较器实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-01-01
  • 一文搞懂Java并发AQS的共享锁模式

    一文搞懂Java并发AQS的共享锁模式

    这篇文章主要为大家阐述AQS另外一个重要模式,共享锁模式。共享锁可以由多个线程同时获取, 比较典型的就是读锁,感兴趣的小伙伴可以了解一下
    2022-10-10
  • 基于 Cursor 开发 Spring Boot 项目详细攻略

    基于 Cursor 开发 Spring Boot 项目详细攻略

    Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配置、代码生成、数据库连接及部署,提供智能辅助、代码优化与错误修复功能,助力高效开发,本文给大家介绍基于Cursor开发Spring Boot项目详细攻略,感兴趣的朋友一起看看吧
    2025-09-09
  • Java基础之面向对象机制(多态、继承)底层实现

    Java基础之面向对象机制(多态、继承)底层实现

    这篇文章主要介绍了Java基础之面向对象机制(多态、继承)底层实现,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • Java将不同的List集合复制到另一个集合常见的方法

    Java将不同的List集合复制到另一个集合常见的方法

    在Java中,有时候我们需要将一个List对象的属性值复制到另一个List对象中,使得两个对象的属性值相同,这篇文章主要介绍了Java将不同的List集合复制到另一个集合常见的方法,需要的朋友可以参考下
    2024-09-09
  • Java设计模式之创建者模式详解

    Java设计模式之创建者模式详解

    这篇文章主要介绍了Java设计模式之创建者模式详解,创建者模式,顾名思义,就是提供友好的创建对象的方式 ,对象都是 new 出来的,但是在一些情况下,这种方式不是很友好,首先,它不够直观,需要的朋友可以参考下
    2023-08-08
  • Java实现按年月打印日历功能【基于Calendar】

    Java实现按年月打印日历功能【基于Calendar】

    这篇文章主要介绍了Java实现按年月打印日历功能,涉及java基于Calendar进行日期运算的相关操作技巧,需要的朋友可以参考下
    2018-03-03
  • SpringBoot + Vue + ElementUI 实现 el-table 分页功能(详细步骤)

    SpringBoot + Vue + ElementUI 实现 el-table 分页功能(详细步骤)

    本文详细介绍了使用SpringBoot和Vue.js结合ElementUI实现分页功能的数据表格,从后端分页逻辑到前端展示和状态管理,全面解析如何高效处理大量数据,提升用户体验与系统性能,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • springboot项目中全局设置用UTC+8

    springboot项目中全局设置用UTC+8

    本文主要介绍了springboot项目中全局设置用UTC+8,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • 一文了解Seata的实现原理

    一文了解Seata的实现原理

    随着业务发展,单体系统逐渐无法满足业务的需求,分布式架构逐渐成为大型互联网平台首选。伴随而来的问题是,本地事务方案已经无法满足,分布式事务相关规范和框架应运而生。本文主要介绍Seata的实现原理
    2021-06-06

最新评论