Java8-HashMap实现原理

news2025/1/11 2:47:18

目录

HashMap原理

hashmap的put流程:

HashMap扩容机制:

HashMap的寻址算法:

HashMap原理

HashMap的底层数据结构是由,数组,链表和红黑树组成的。

当我们往HashMap中put元素的时候,利用key的hashCode重新计算hash出当前对象的元素在数组中的下标。

当然这样也可能会出现一个问题,如果出现hash值相同的key,那么如何进行处理呢?有的人可能会问,哈希值怎么可能相同,那么此时我们需要明白一个道理,很简单,鸽巢原理:可以概括为一句话,假如我们有10个鸽巢,11个鸽子,那么此时必定至少有一个鸽巢内的鸽子数是大于两个的。应用到Hashmap中,如果存储的Key的数量大于数组的原始长度,势必会存在hash值相同的key。那么此时就需要引入链表了。

在解决上述问题时,HashMap给出了链表的结构来进行存储,如果出现hash值相同的key,那么此时会出现两种情况。

1.如果key相同,则覆盖原始key对应的value;

2.如果key不同,那么就将当前的key-value放入到链表(尾插法)或者红黑树中;

在获取时,会直接找到hash对应的下标,再进一步判断key值是否相同,从而找到对应值。

JDK1.8在解决hash冲突时,当链表长度大于阈值时,并且数组长度达到64,就会将链表转成红黑树,以较少搜索时间。扩容resize()时,红黑树拆分成的树节点小于等于临界值6时,则退化成链表。

hashMap默认的加载因子为0.75,默认的初始容量为16.当存储的阈值为大于16*0.75时会自动进行扩容

hashmap的put流程:

可以分为以下几步:

1.会判断table是否为空,如果为空,则代表是第一次进行创建,于是会初始化一个长度为16的数组

2.然后根据你的key计算索引值,如果该索引位置为null,那么就直接插入,否则就去判断该key是否存在在该位置。

3.如果是存在,那么直接覆盖,如果不存在,那么就会判断该位置是否为红黑树,走红黑树的添加逻辑。

4.如果不是红黑树,那么就说明此处是一个链表,那么就遍历该链表,判断key是否存在在该链表之中,如果存在直接覆盖。

5.如果不存在,那么会在链表尾部进行插入,此时还需判断链表是否长度大于8,如果大于则转成红黑树,否则直接返回。

6.插入完成之后还会判断整个table的size是否大于扩容的阈值,如果大于的话,就会对hashmap进行扩容。

HashMap扩容机制:

1.首先会判断你的capacity是否是大于0的如果不大于0说明你是第一次来,那么就进行设置数组容量为16,负载因子为0.75,阈值为12.

2.如果不是第一次来,那么就会将hashmap大小设置为两倍的capacity,然后去新建数组,此时他会去遍历旧的数组,对原来链表里的key重新进行添加。

3.如果该位置为空,那么直接就添加,

4.如果该位置为红黑树,那么就是红黑树的添加逻辑。

5.如果该位置为链表,那么就会遍历链表,然后让新的hash值和原来的链表长度按位与,如果值为0那么新表的位置也为原来的位置,否则就是原来的位置加原来的数组容量,比如原来是3位置存储,大小为16,那么新的位置就是19。

HashMap的寻址算法:

1.计算对象的hashcode

2.调用hash()方法二次哈希,hashcode右移16位再异或运算,让哈希分布更均匀。

3.再使用(capacity-1)& oldcapacity得到索引。

使用2的n次方作为哈希的数组长度,可以代替取模运算,效率更高。

另一个原因就是,如果计算索引的时候,可以让hash& oldcap == 0 的值留在原来的位置,否则让新位置=旧位置+oldcap。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

HC32F103BCB使用SPI获取AS5040编码器数据

1.AS5040介绍 2.硬件电路 硬件上使用SSI通信方式连接。 3.配置硬件SPI 查看手册,AS5040时序 可以看到在空闲阶段不发生数据传输的时候时钟(CLK)和数据(DO)都保持高电位(tCLKFE阶段),在第一个脉冲的下降沿触发编码器载入发送数据,然后每一个…

【Unity Shader入门精要 第9章】更复杂的光照(四)

1. 透明度测试物体的阴影 对于物体有片元丢弃的情况,比如透明度测试或者后边会讲到的消融效果,使用默认的 ShadowCaster Pass 会产生问题,这是因为该Pass在生成阴影映射纹理时,没有考虑被丢弃的片元,而是使用完整的模…

FTP文件传输议

FTP是一种文件传输协议:用来上传和下载,实现远程共享文件,和统一管理文件 工作原理:用于互联网上的控制文件的双向传输是一个应用程序。工作在TCP/IP协议簇的,其传输协议是TCP协议提高文件传输的共享性和可靠性&#…

阅读笔记——《AFLNeTrans:状态间关系感知的网络协议模糊测试》

【参考文献】洪玄泉,贾鹏,刘嘉勇.AFLNeTrans:状态间关系感知的网络协议模糊测试[J].信息网络安全,2024,24(01):121-132.【注】本文仅为作者个人学习笔记,如有冒犯,请联系作者删除。 目录 摘要 1、引言 2、背景及动机 2.1、网络协议实现程…

正点原子LWIP学习笔记(二)MAC简介

MAC简介 一、MAC简介(了解)二级目录三级目录 二、ST的ETH框架(了解)三、SMI站管理接口(熟悉)四、介质接口MII、RMII(熟悉) 一、MAC简介(了解) STM32 的 MAC …

Ubuntu24.04设置静态IP地址

Ubuntu24.04设置静态IP地址 前言:vm17.5的动态IP问题 第一个是设置的静态IP我们可以看到是forever,第二个则是动态IP则是一天的时间。 如果我们不设置静态IP的话,那么可能在本地测试项目的时候,第二天发现一些服务不能用了&#…

13.js对象

定义 一种复杂数据类型,是无序的(不保留键的插入顺序),以键值对({key:value})形式存放的数据集合 对象的创建 (1)字面量创建 var 对象名{ } (2)内部构造函数创建 v…

VirtualBox安装ubuntu22.04记录

一,VirtualBox 软件安装 虚拟机(Virtual Machine)指通过软件模拟的具有完整硬件系统功能的、运行在一个完全隔离环境中的完整计算机系统。在实体计算机中能够完成的工作在虚拟机中都能够实现。 常见的虚拟机软件主要有两款 VMware 和 VirtualBox 。VMwar…

争议湖北消费金融2023年业绩,营收下滑or财务内控重大缺陷?

近日,湖北消费金融股份有限公司(下称“湖北消费金融”)披露了2023年度相关信息,包括股权结构、关联方、董事会、分支机构、资产负债情况等信息。 据介绍,湖北消费金融的注册资本为10.058亿元,法定代表人为…

linux---线程控制

线程和进程 以前我们要同时跑多个程序,可以通过fork()多个子进程,然后通过系统函数进行程序的替换,但是创建进程代价大,不仅要拷贝一份父进程的地址空间,页表,文件表述符表等。但是线程不需要因为是进程的…

使用JavaScript日历小部件和DHTMLX Gantt的应用场景(三)

DHTMLX Suite UI 组件库允许您更快地构建跨平台、跨浏览器 Web 和移动应用程序。它包括一组丰富的即用式 HTML5 组件,这些组件可以轻松组合到单个应用程序界面中。 DHTMLX Gantt是用于跨浏览器和跨平台应用程序的功能齐全的Gantt图表,可满足项目管理应用…

【调试笔记-20240520-Linux-在 WSL2 / Ubuntu 20.04 中编译 QEMU 可运行的 OVMF 固件】

调试笔记-系列文章目录 调试笔记-20240520-Linux-在 WSL2 / Ubuntu 20.04 中编译 QEMU 可运行的 OVMF 固件 文章目录 调试笔记-系列文章目录调试笔记-20240520-Linux-在 WSL2 / Ubuntu 20.04 中编译 QEMU 可运行的 OVMF 固件 前言一、调试环境操作系统:Windows 10 …

作业-day-240523

思维导图 知识点问答 1、IO多路复用的原理 1、创建一个检测文件描述符的容器 fd_set fds; 2、将需要检测的文件描述符放入容器中 FD_SET(文件描述符,&fds); 3、通过一个阻塞函数阻塞等待容器中是否有事件产生,如果有一个或多个事件产生&#xff0c…

C++设计模式|结构型 适配器模式

1.什么是适配器模式? 可以将⼀个类的接⼝转换成客户希望的另⼀个接⼝,主要⽬的是 充当两个不同接⼝之间的桥梁,使得原本接⼝不兼容的类能够⼀起⼯作。 2. 适配器模式的组成 (1)接口类,给客户端调用&…

软件设计师-上午题-计算题汇总

一、存储系统 - 存储容量计算(字节编址、位编址、芯片个数) 内存地址是16进制 内存地址编址的单位是Byte,1K1024B 1B 8 bit 1.计算存储单元个数 存储单元个数 末地址 - 首地址 1 eg. 按字节编址,地址从 A4000H 到 CBFFFH&…

汽车IVI中控开发入门及进阶(十八):显示技术之Frame Buffer帧缓冲器

Frame Buffer帧缓冲器(帧缓冲器,有时是帧存储器)是随机存取存储器(RAM)的一部分,包含驱动视频显示器的位图。它是一个内存缓冲区,包含表示完整视频帧中所有像素的数据。现代视频卡的核心包含帧缓冲电路。该电路将内存中的位图转换为可以在计算机监视器上显示的视频信号。…

Redis未授权访问漏洞复现 CNVD-2019-21763 CNVD-2015-07557

CNVD-2019-21763 漏洞描述 Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API。 由于在Reids 4.x及以上版本中新增了模块功能,攻击者可通过外部拓展,在Redis中实现一个新…

redis集群不允许操作多个key解决方案、redis key负载均衡方案

前提 在cluster redis 中进行同一个命令处理不同的key会报错:CROSSSLOT Keys in request dont hash to the same slot,例如: 此示例使用sdiff 命令对pool_1与pool_2进行diff操作。 那么我们在业务场景中就需要将集群redis中的不同key进行操作,我们该如何处理呢? 本次的…

多线程(八)

一、wait和notify 等待 通知 机制 和join的用途类似,多个线程之间随机调度,引入 wait notify 就是为了能够从应用层面上,干预到多个不同线程代码的执行顺序.( 这里说的干预,不是影响系统的线程调度策略 内核里的线程调度,仍然是无序的. 相当于是在应用程序…

探秘QT 5软件开发:从零到实战的旅程

新书上架~👇全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我👆,收藏下次不迷路┗|`O′|┛ 嗷~~ 目录 一、QT 5软件开发的启航 二、GUI框架比较与QT 5的选择 三、QT 5环境搭建与软件设计实战 四…