返回顶部
首页 > 资讯 > 精选 >对HashMap的思考及手写实现
  • 439
分享到

对HashMap的思考及手写实现

2023-06-02 18:06:44 439人浏览 泡泡鱼
摘要

作者:张丰哲原文:https://www.jianshu.com/p/b638f19aeb64HashMap是Java中常用的集合,而且HashMap的一些思想,对于我们平时解决业务上的一些问题,在思路上有帮助,基于此,本篇博客将分析Has

作者:张丰哲原文:https://www.jianshu.com/p/b638f19aeb64

HashMap是Java中常用的集合,而且HashMap的一些思想,对于我们平时解决业务上的一些问题,在思路上有帮助,基于此,本篇博客将分析HashMap底层设计思想,并手写一个迷你版的HashMap!

对HashMap的思考及手写实现

对HashMap的思考

对HashMap的思考及手写实现

HashMap底层数据结构

第一,如图所示,HashMap有3个要素:hash函数+数组+单链表

第二,对于hash函数而言,需要考虑些什么?

要快,对于给定的Key,要能够快速计算出在数组中的index。那么什么运算够快呢?显然是位运算!

要均匀分布,要较少碰撞。说白了,我们希望通过hash函数,让数据均匀分布在数组中,不希望大量数据发生碰撞,导致链表过长。那么怎么办到呢?也是利用位运算,通过对数据的二进制的位进行移动,让hash函数得到的数据散列开来,从而减低了碰撞的概率。

如果发生了碰撞怎么办?上面的图其实已经说明了jdk的HashMap是如何处理hash冲突的,就是通过单链表解决的。那么除了这个方法,还有其他思路么?比如说,如果发生冲突,那么记下这个冲突的位置为index,然后在加上固定步长,即index+step,找到这个位置,看一下是否仍然冲突,如果继续冲突,那么按照这个思路,继续加上固定步长。其实这就是所谓的线性探测来解决Hash冲突的方法!

通过写一个迷你版的HashMap来深刻理解

定义接口

对HashMap的思考及手写实现

接口

定义一个接口,对外暴露快速存取的方法。

注意MyMap接口内部定义了一个内部接口Entry。

接口实现

对HashMap的思考及手写实现

MyHashMap定义

HashMap的要素之一,就是数组,自然在这里,我们要定义数组,数组的初始化大小,还要考虑扩容的阀值。

看MyHashMap的构造

对HashMap的思考及手写实现

构造方法

构造方法有什么好说的呢?

仔细观察下,你会发现,其实这里使用到了“门面模式”。这里的2个构造方法其实指向的是同一个,但是对外却暴露了2个“门面”!

Entry

对HashMap的思考及手写实现

Entry

HashMap的要素之一,单链表的体现就在这里!

看put如何实现

对HashMap的思考及手写实现

put

第一,要考虑是否扩容?

HashMap中的Entry的数量(数组以及单链表中的所有Entry)是否达到阀值?

第二,如果扩容,意味着新生成一个Entry[],不仅如此还得重新散列。

第三,要根据Key计算出在Entry[]中的位置,定位后,如果Entry[]中的元素为null,那么可以放入其中,如果不为空,那么得遍历单链表,要么更新value,要么形成一个新的Entry“挤压”单链表!

hash函数

对HashMap的思考及手写实现

MyHashMap提供的hash函数

对HashMap的思考及手写实现

JDK的HashMap提供的hash函数

我这里参考了JDK的HashMap的hash函数的实现,这里也再次说明了:要想散列均匀,就得进行二进制的位运算!

resize和rehash

对HashMap的思考及手写实现

resize/rehash

这里可以看出,对于HashMap而言,如果频繁进行resize/rehash操作,是会影响性能的。

resize/rehash的过程,就是数组变大,原来数组中的entry元素一个个的put到新数组的过程,需要注意的是一些状态变量的改变。

get实现

对HashMap的思考及手写实现

get

get很简单,只需要注意在遍历单链表的过程中使用== or equals来判断下即可。

Test测试

对HashMap的思考及手写实现

利用MyHashMap进行存取

运行结果

对HashMap的思考及手写实现

result

OK,一个迷你版的HashMap就写好了,你学到了么?

--结束END--

本文标题: 对HashMap的思考及手写实现

本文链接: https://lsjlt.com/news/231018.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 对HashMap的思考及手写实现
    作者:张丰哲原文:https://www.jianshu.com/p/b638f19aeb64HashMap是Java中常用的集合,而且HashMap的一些思想,对于我们平时解决业务上的一些问题,在思路上有帮助,基于此,本篇博客将分析Has...
    99+
    2023-06-02
  • HashMap原理及手写实现部分区块链特征
    目录写在前面JDK7和JDK8中的HashMap正文写在前面 最近有很多的粉丝私信我,说自己在面试的时候,老是被人问HashMap的原理,但是在实际的工作中,也只是使用HashMap...
    99+
    2024-04-02
  • 关于读写锁算法的Java实现及思考是怎么样的
    本篇文章给大家分享的是有关关于读写锁算法的Java实现及思考是怎么样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。问题背景:多个线程对一个共享的资源进行读写访问。写线程之间需...
    99+
    2023-06-17
  • JavaScript面试必考之实现手写Promise
    目录Promise手写框架完整代码测试resolverejectPromise手写 Promise作为面试必考题,Promise的手写也是面试官必问的问题,所以对于Promise我们...
    99+
    2022-12-08
    JavaScript实现手写Promise JavaScript手写Promise JavaScript Promise
  • 手写mybatis完整sql插件问题及实现思路
    问题产生 我们在使用mybatis的过程中,如果开启了mysql的日志功能的话,会在控制台打印一些sql的信息,但是日志中的sql语句,是没有拼接参数的,也就是说,是不可以直接放到数...
    99+
    2024-04-02
  • vue 长列表数据刷新的实现及思考
    目录开篇一、效果展示二、代码开篇 通过 vue 进行列表展示的时候如果数据太多可能会卡顿,这里通过滑动计算只创建跟刷新可见部分 dom 元素,这里仅仅代表着复用思路 一、效果展示 ...
    99+
    2023-05-14
    vue 长列表数据刷新 vue 数据刷新
  • 对于组件库的思考及技术梳理详解
    目录为什么要做?组件库的技术选型组件库的方向组件库的设计结语为什么要做? 18年的时候觉得写ui组件库的人都好牛,出于好奇看了elementui、iview和Vant的源码。从那之...
    99+
    2023-02-01
    组件库技术梳理 组件库思考
  • springboot实现拦截器的3种方式及异步执行的思考
    目录springboot 拦截器springboot 入门案例maven 引入启动类定义 Controller拦截器定义基于 Aspect基于 HandlerInterceptor基...
    99+
    2024-04-02
  • ASP.NET如何实现团队分工的思考
    这篇文章主要介绍“ASP.NET如何实现团队分工的思考”,在日常操作中,相信很多人在ASP.NET如何实现团队分工的思考问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”ASP.NET如何实现团队分工的思考”的疑...
    99+
    2023-06-17
  • 绝对定位参考方法的实现
    实现绝对定位的参照方法,需要具体代码示例 随着Web开发的不断发展,对于页面布局的要求也越来越高。绝对定位是一种常用的布局方式,可以精确地指定元素在页面中的位置。本文将介绍如何通过CSS来实现绝对定位,并提供具体的代码示例。 一...
    99+
    2024-01-23
    实现 绝对定位 参照方法
  • 利用Java如何实现对HashMap的集合使用
    这期内容当中小编将会给大家带来有关利用Java如何实现对HashMap的集合使用,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。HashMap是最常用的Map集合,它的键值对在存储时要根据键的哈希码来确定值...
    99+
    2023-05-31
    java 集合 hashmap
  • 手写实现JS中的new
    目录1 new 运算符简介2 new 究竟干了什么事3 模拟实现 new 运算符4 补充⚠ 预备知识: 了解原型和原型链 了解this绑定 1 new...
    99+
    2024-04-02
  • 在Java中使用HashMap来实现访问的键值对
    今天就跟大家聊聊有关在Java中使用HashMap来实现访问的键值对,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Map是日常编程中比较常用的数据结...
    99+
    2024-04-02
  • 不容错过的HashMap实现原理及源码分析
    哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,而HashMap的实现原理也常常出现在各类的面试题中,重要性可见一斑。本文...
    99+
    2023-06-02
  • 手写简版kedis分布式key及value服务的实现及配置
    目录前言rocksdb特征RestExpress实现kedis创建服务并绑定端口创建RocksDB引擎api操作类设置请求路由启动插入数据获取数据文末结语前言 今天博主主要介绍两个开...
    99+
    2024-04-02
  • (手写)PCA原理及其Python实现图文详解
    目录1、背景2、样本均值和样本方差矩阵3、PCA3.1 最大投影方差3.2 最小重构距离4、Python实现总结1、背景 为什么需要降维呢? 因为数据个数 N 和每个数据的维度 p 不满足 N >> p...
    99+
    2022-06-02
    Python PCA原理
  • Python手写回归树的实现
    目录回归树创建子节点预测计算误差概括的步骤更深入的模型在本篇文章中,我们将介绍回归树及其基本数学原理,并从头开始使用Python实现一个完整的回归树模型。 为了简单起见这里将使用递归...
    99+
    2024-04-02
  • Java中​HashMap的工作原理及实现方法是什么
    今天小编给大家分享一下Java中HashMap的工作原理及实现方法是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。Has...
    99+
    2023-06-03
  • Java手写Redis服务端的实现
    目录零,起因一,redis通讯与Netty1,tcp2,协议3,编解码4,命令处理二,redis的数据结构1,底层主结构2,key3,list4,set5,hash6,zset三,r...
    99+
    2024-04-02
  • JDK动态代理过程原理及手写实现详解
    目录JDK动态代理的过程手写实现JDK动态代理创建MyInvocationHandler接口创建MyClassLoader类加载器创建代理类使用自定义动态代理类创建接口创建被代理接口...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作