迹忆客 专注技术分享

当前位置:主页 > 学无止境 > 编程语言 > Java >

Java中 HashMap 的工作原理

作者:迹忆客 最近更新:2023/05/09 浏览次数:

什么是哈希

哈希是将对象转换为整数值的过程。 整数值有助于索引和更快的搜索。

什么是 HashMap

HashMap 是 Java 集合框架的一部分。 它使用一种称为散列的技术。 它实现了 Map 接口。 它将数据存储在 Key 和 Value 对中。 HashMap 包含一个节点数组,节点表示为一个类。 它在内部使用数组和 LinkedList 数据结构来存储 Key 和 Value。 HashMap 中有四个字段。

 

Java 节点表示
Java 节点表示

 

在了解 HashMap 的内部工作原理之前,必须了解 hashCode() 和 equals() 方法。

  • equals() :检查两个对象的相等性。 它比较 key,无论它们是否相等。 它是 Object 类的一个方法。 它可以被覆盖。 如果重写 equals() 方法,则必须重写 hashCode() 方法。
  • hashCode() :这是 object 类的方法。 它以整数形式返回对象的内存引用。 从方法接收到的值用作桶号。 桶号是 Map 内元素的地址。 null Key 的哈希码为 0。
  • Buckets :节点的数组称为桶。 每个节点都有一个类似于 LinkedList 的数据结构。 多个节点可以共享同一个桶。 容量可能不一样。

 

Bucket 中分配节点
Bucket 中分配节点

 

在 HashMap 中插入键、值对

我们使用 put() 方法在 HashMap 中插入 Key 和 Value 对。 HashMap 的默认大小为 16(0 到 15)。

示例

在下面的示例中,我们要在 HashMap 中插入三个 (Key, Value) 对。

HashMap<String, Integer> map = new HashMap<>();  
map.put("Aman", 19);  
map.put("Sunny", 29);  
map.put("Ritesh", 39);  

让我们看看 Key, value 对将在哪个索引处保存到 HashMap 中。 当我们调用 put() 方法时,它会计算 Key“Aman”的哈希码。 假设“Aman”的哈希码是2657860。要将Key存储在内存中,我们必须计算索引。

计算索引

索引最小化数组的大小。 指数计算公式为:

Index = hashcode(Key) & (n-1)  

其中 n 是数组的大小。 因此,“Aman”的索引值为:

Index = 2657860 & (16-1) = 4  

值 4 是计算的索引值,其中 Key 和 value 将存储在 HashMap 中。

 

Java 中HashMap工作原理
Java 中HashMap工作原理

 


哈希冲突

当计算的索引值对于两个或多个 Key 相同时就是这种情况。 让我们计算另一个键“Sunny”的哈希码。 假设“Sunny”的哈希码是63281940。要将Key存储在内存中,我们必须使用索引公式计算索引。

Index=63281940 & (16-1) = 4  

4 是我们计算出来的索引值,其中 Key 将存储在 HashMap 中。 在这种情况下,equals() 方法检查两个 Key 是否相等。 如果 Keys 相同,则将值替换为当前值。 否则,通过 LinkedList 将此节点对象连接到现有节点对象。 因此,两个 Key 都将存储在索引 4 处。

 

Java HashMap 哈希冲突
Java HashMap 哈希冲突

 

同样,我们将存储key “Ritesh”。 假设 Key 的哈希码为 2349873。索引值为 1。因此,此 Key 将存储在索引 1 处。

Java HashMap 哈希冲突解决


HashMap 中 get() 方法

get() 方法用于通过其 Key 获取值。 如果不知道key,它将不会获取值。 调用 get(K Key) 方法时,会计算 Key 的哈希码。

假设我们必须获取 Key “Aman”。 将调用以下方法。

map.get(new Key("Aman"));  

它生成哈希码为2657860。现在使用索引公式计算2657860的索引值。 正如我们上面计算的那样,索引值为 4。 get() 方法搜索索引值 4。它将第一个元素 Key 与给定的 Key 进行比较。 如果两个键相等,则返回值,否则检查节点中的下一个元素是否存在。 在我们的场景中,它是节点的第一个元素并返回值 19。

让我们获取另一个 key “Sunny”。

“Sunny”键的哈希码是63281940。计算出来的63281940的索引值为4,正如我们为put()方法计算的那样。 转到数组的索引 4 并将第一个元素的 Key 与给定的 Key 进行比较。 它还比较键。 在我们的场景中,给定的 Key 是第二个元素,节点的 next 为 null。 它将第二个元素 Key 与指定的 Key 进行比较,并返回值 29。如果节点的下一个为 null,则返回 null。

转载请发邮件至 1244347461@qq.com 进行申请,经作者同意之后,转载请以链接形式注明出处

本文地址:

相关文章

在 Java 中获取文件大小

发布时间:2023/05/01 浏览次数:139 分类:Java

Java 提供了不同的方法来获取文件的字节大小。 本教程演示了在 Java 中获取文件大小的不同方法。使用 Java IO 的文件类获取文件大小 Java IO 包的 File 类提供了以字节为单位获取文件大小的功能。

Java 中的文件分隔符

发布时间:2023/05/01 浏览次数:108 分类:Java

本篇文章介绍了 Java 中的文件分隔符。Java 中的文件分隔符 文件分隔符是用来分隔目录的字符; 例如,Unix 使用 /,Windows 使用 \ 作为文件分隔符。

Java 中的文件过滤器

发布时间:2023/05/01 浏览次数:193 分类:Java

本篇文章介绍如何在 Java 中使用 FileFilter。FileFilter 用于过滤具有特定扩展名的文件。 Java内置包IO和Apache Commons IO为FileFilter提供了类和接口来进行文件过滤操作。

Java 获取 ISO 8601 格式的当前时间戳

发布时间:2023/05/01 浏览次数:132 分类:Java

本篇文章介绍了 ISO 8601 日期格式、其重要性及其在 Java 中的使用。 它还列出了一些优点来强调为什么应该使用 ISO 格式来表示日期。

在 Java 中获取数组的子集

发布时间:2023/05/01 浏览次数:142 分类:Java

本篇文章介绍了几种在 Java 中获取数组子集的方法。使用 Arrays.copyOf() 方法获取数组的子集 使用 Arrays.copyOfRange() 方法获取数组的子集

用 Java 填充二维数组

发布时间:2023/05/01 浏览次数:110 分类:Java

二维数组是基于表结构的,即行和列,填充二维数组不能通过简单的添加到数组操作来完成。 本篇文章介绍如何在 Java 中填充二维数组。

Java 中的自然排序

发布时间:2023/05/01 浏览次数:132 分类:Java

Java 中最常用的顺序是自然顺序。 本文将展示如何使用 naturalOrder() 函数对数组进行排序。

计算 Java 数组中的重复元素

发布时间:2023/05/01 浏览次数:202 分类:Java

本篇文章介绍Java计算数组中重复元素的方法。计算 Java 数组中的重复元素。我们可以创建一个程序来计算数组中的重复元素。 该数组可以是未排序的,也可以是已排序的。

Java 中 List 和 Arraylist 的区别

发布时间:2023/05/01 浏览次数:90 分类:Java

表示为单个单元的一组单个对象称为集合。 在 Java 中,Collection 是一个具有多个已定义接口和类的框架,用于将一组对象表示为一个单元。 它允许我们操纵

扫一扫阅读全部技术教程

社交账号
  • https://www.github.com/onmpw
  • qq:1244347461

最新推荐

教程更新

热门标签

扫码一下
查看教程更方便