hashmap value占用空间大小_HashMap的put和get实现原理及源码分析

14a644dba442e74059f990a2d7535742.png
水平有限,难免会有疏漏之处,如有错误,还请指出,感谢!

前言

HashMa是Java中最常用的集合类框架,也是Java语言中非常典型的数据结构,同时也是我们需要掌握的数据结构,更重要的是进大厂面试必问之一。

98932f78c75f5f470f0df00fd63afbcb.png

数组特点

  • 存储区间是连续,且占用内存严重,空间复杂也很大,时间复杂为O(1)。
  • 优点:是随机读取效率很高,原因数组是连续(随机访问性强,查找速度快)。
  • 缺点:插入和删除数据效率低,因插入数据,这个位置后面的数据在内存中要往后移的,且大小固定不易动态扩展。

链表特点

  • 区间离散,占用内存宽松,空间复杂度小,时间复杂度O(N)。
  • 优点:插入删除速度快,内存利用率高,没有大小固定,扩展灵活。
  • 缺点:不能随机查找,每次都是从第一个开始遍历(查询效率低)。

哈希表特点

以上数组和链表,大家都知道各自优缺点。那么我们能不能把以上两种结合一起使用,从而实现查询效率高和插入删除效率也高的数据结构呢?答案是可以滴,那就是哈希表可以满足,接下来我们一起复习HashMap中的put()和get()方法实现原理。

d1c1be1e374d168f1d28e91dd70d4fea.png

HashMap的put()和get()的实现

1、map.put(k,v)实现原理

第一步:首先将k,v封装到Node对象当中(节点)
第二步:通过哈希算法计算出当前key的hash值
第三步:再通过哈希表函数/哈希算法,将hash值转换成数组的下标,下标位置上如果没有任何元素,就把Node添加到这个位置上。
如果说下标对应的位置上有链表。此时,就会拿着k和链表上每个节点的k进行equal。
如果所有的equals方法返回都是false,那么这个新的节点将被添加到链表的末
尾。如其中有一个equals返回了true,那么这个节点的value将会被覆盖。
// 存储时:
// 这个hashCode方法这里不详述,只要理解每个key的hash是一个固定的int值即可
int hash = key.hashCode(); 
int index = hash % Entry[].length;
Entry[index] = value;

2、map.get(k)实现原理

//数组长度减1与运算hash值
first = tab[(n - 1) & hash]
第一步:先调用k的hashCode()方法得出哈希值,并通过哈希算法转换成数组的下标。
第二步:通过上一步哈希算法转换成数组的下标之后,在通过数组下标快速定位到某个位置上。
重点理解
如果这个位置上什么都没有,则返回null。
如果这个位置上有单向链表,那么它就会拿着参数K和单向链表上的每一个节点的K进行equals,如果所有equals方法都返回false,则get方法返回null。
如果其中一个节点的K和参数K进行equals返回true,那么此时该节点的value就是我们要找的value了,get方法最终返回这个要找的value。

3、为何随机增删、查询效率都很高的原因是?

原因:增删是在链表上完成的,而查询只需扫描部分,则效率高。

HashMap集合的key,会先后调用两个方法,hashCode and equals方法,这两个方法都需要重写。

4、为什么放在hashMap集合key部分的元素需要重写equals方法?

因为equals默认比较是两个对象内存地址

5、HashMap总结

  • 无序,不可重复
为什么是无序的?
  • 因为不一定挂到哪一个单向链表上的,因此加入顺序和取出也不一样。
怎么保持不可重复?
  • 使用equals方法来保证HashMap集合key不可重复,如key重复来,value就会覆盖。存放在HashMap集合key部分的元素,其实就是存放在HashSet集合中,则HashSet集合也需要重写equals和hashCode方法。
  • hashmap集合的默认初始化容量为16,默认加载因子为0.75,也就是说这个默认加载因子是当hashMap集合底层数组的容量达到75%时,数组就开始扩容。
  • hashmap集合初始化容量是2的陪数,为了达到散列均匀,提高hashmap集合的存取效率,

6、注意JDK8之后

JDK8之后,如果哈希表单向链表中元素超过8个,那么单向链表这种数据结构会变成红黑树数据结构。当红黑树上的节点数量小于6个,会重新把红黑树变成单向链表数据结构,官方源码如下图。

21ee8a22460ae90a0fbcc85c1cf3205c.png

问题:

如果O1和O2的hash值相同,就会存放到同一个单向链表上,

如果不同,但由于哈希算法执行结束之后转换的数组下标可能相同,此时会发上“哈希碰撞”。

HashMap的存取是采用什么算法实现?

// 存储时:
// 这个hashCode方法这里不详述,只要理解每个key的hash是一个固定的int值即可
int hash = key.hashCode(); 
int index = hash % Entry[].length;
Entry[index] = value;// 取值时:
int hash = key.hashCode();
int index = hash % Entry[].length;
return Entry[index];

7、高频面试题

  • HashMap的工作原理是什么?
  • HashMap中的“死锁”是怎么回事?
  • HashMap中能put两个相同key吗?为什么?
  • HashMap中的键值可以为空吗?原理?
  • HashMap扩容机制?

另,如果觉得这本篇文章写得不错,有点东西的话,记得来个三连【点赞+关注+分享】。
需要大数据、Java、redis、Dubbo框架等教程,关注微信公众号:自学大数据踩的抗 【回复相关术语】

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

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

相关文章

pthread_cond_wait的spurious wakeup问题

最近在温习pthread的时候,忽然发现以前对pthread_cond_wait的了解太肤浅了。昨晚在看《Programming With POSIX Threads》的时候,看到了pthread_cond_wait的通常使用方法: pthread_mutex_lock(); while(condition_is_false) pthread_con…

CABasicAnimation

CABasicAnimation 自己只有三个property fromValue toValue ByValue 当你创建一个 CABasicAnimation 时,你需要通过-setFromValue 和-setToValue 来指定一个开始值和结束值。 当你增加基础动画到层中的时候,它开始运行。当用属性做动画完成时,例如用位置属性做动画,层就会立…

python中func函数用法_python之4类回调函数的使用方法

原标题:python之4类回调函数的使用方法 将函数作为参数传递给另一个函数,一共分为4种情况: 将普通函数传递给普通函数 将普通函数传递给类成员函数 将类成员函数传递给普通函数 将类成员函数传递给类成员函数 这4种情况,在python中…

c#sql防注入模糊查询_SQL中利用LIKE实现模糊查询的功能

大家好,今日继续讲解《VBA数据库解决方案》,今日讲解的内容是:利用ADO,实现模糊查询。在上一讲中,我们实现了利用ADO快速查找的功能,今日我们实现工作表中模糊查找的功能。我们仍是利用上一讲的数据实现, 在"两表…

C++的一般引用及其数组引用

引用就是某一变量(目标)的一个别名,对引用的操作与对变量直接操作完全一样。 引用的声明方法:类型标识符 &引用名目标变量名; 【例1】:int a; int &raa; //定义引用ra,它是变量a的引用&#xff0…

马云卸任CEO演讲全文:明天起生活将是我的工作

马云:大家晚上好!谢谢各位,谢谢大家从全国各地,我知道也有从美国、英国和印度来的同事,感谢大家来到杭州,感谢大家参加淘宝的十周年! 今天是一个非常特别的日子,当然对我来讲&#x…

idea断点_IDEA Debug 无法进入断点的解决方法

前言某个多模块项目中使用多个版本的 Spring,如 Spring 4,Spring 5,在使用 IDEA Debug 过程中发现,Spring 部分 jar 如 spring-core 中的上面断点,IDEA 可以成功进入。但是有部分如 spring-context IDEA 始终无法进入断…

win10taskkill无法终止进程_Win10无法终止进程拒绝访问

用任务管理器强制结束一些已经不使用程序的进程,是很多用户会用的功能之一,但是最近有使用win10系统的用户,遇到结束进程的时候,被拒绝访问。遇到这样的问题,给大家带来了这篇文章的方法,希望能帮助到大家。…

c++对数组的引用

所谓数组引用,即指向数组的引用;如 int a[10] ; int (&b)[10] a ;如果写成 int a[10] ;int* &b a ;将会报错: cannot convert from int [10] to int *&。或许你会说在数组名不就是指向这个数组的一个指针吗&#…

python 线程池_老程序员的经验分享:Python 从业十年是种什么体验?

出于某些原因,想记录一下我过去数年使用 Python 的经验和一些感悟。毕竟算是一门把我带入互联网行业的语言,而我近期已经几乎不再写 Py 代码, 做一个记录,也许会对他人起到些微的帮助,也算是纪念与感恩了。作者&#x…

【转】C#中Invoke的用法

在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,Invoke 和 BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。正确的做法是将工作线程中涉及…

练字格子纸模板pdf_高考英语作文模板(总结八种常考题型,配合例文,纯手打的)...

又是一年高考结束,又有不少新高三的学弟学妹问我一些学习上的方法。额,今天我们就单说这个英语作文。英语作文第一件事练字,其次背模板。高考无非就几种信件变着花考察。几种基本信件模板稍加变通就可以很简单完成作文。本人2019年河南考生&a…

GCC 提供的原子操作

gcc从4.1.2提供了__sync_*系列的built-in函数,用于提供加减和逻辑运算的原子操作。其声明如下:type __sync_fetch_and_add (type *ptr, type value, ...) type __sync_fetch_and_sub (type *ptr, type value, ...) type __sync_fetch_and_or (type *ptr,…

google js cdn_「效率工具」模拟CDN的浏览器扩展程序,改善在线隐私

更多互联网新鲜资讯、工作奇淫技巧关注原创【飞鱼在浪屿】(日更新)LocalCDN是一个Web浏览器扩展,它模仿Content Delivery Networks以改善在线隐私。它拦截流量,在本地找到静态资源,然后将其注入环境。所有这些都是自动发生的,因此…

如何保证战略落地_如何让战略落地:流程管理的道法术器让战略落地提升竞争力...

从0开始学管理:专注科学系统提升管理能力:基础 中层 高层 综合管流程革命一、流程理念流程六要素:客户 、活动间的关系 、活动 、输出 、输入 、价值二、流程浮现什么是端到端的流程:业务全程闭环 、从开始到结束 、从发起到完成 …

出口同比中国经济三大怪状折射出啥危机?

题记:写这篇博客要主是加深自己对出口同比的认识和总结实现算法时的一些验经和训教,如果有错误请指出,万分感谢。 与欧美国家经济相比,中国经济形势更加错综庞杂,这不仅仅是因为中国官方颁布的经济数据掺杂水份&#x…

/sys/class/gpio 文件接口操作IO端口(s3c2440)

在嵌入式设备中对GPIO的操作是最基本的操作。一般的做法是写一个单独驱动程序,网上大多数的例子都是这样的。其实linux下面有一个通用的GPIO操作接口,那就是我要介绍的 “/sys/class/gpio” 方式。 首先,看看系统中有没有“/sys/class/gpio”…

elf文件格式_elf文件,readelf

汽车电子开发过程中,代码完成后,程序编译完成 会生成 elf文件 或 hex文件,可以烧录到MCU中调试,那么究竟什么是 elf文件呢? elf 文件中又包含哪些信息? 如何解析 elf文件呢?1. What is elf fileELF(Execut…

是人是谁_其实,我们每个人心中都有一把尺子,谁好谁歹谁心里都明白……

有一些人,对别人有一点好,就能整天挂在嘴边,生怕别人能忘了似的,还有一些人,对谁好,都不喜欢说在嘴上,就愿意那么默默无闻地善良着,把温暖悄悄传递给别人的心灵,这是我们…

一个伟大计划终于完成了(粉丝联盟网正式上线了)

一个伟大的计划是指 搭建一个拥有独立顶级域名的网站。(2009年时,我就有这个想法,今天终于实现了。)网站:http://FansUnion.cn/ ; 粉丝联盟网FansUnion的含义 大一时,我开始玩网络游戏-天龙八部。当时取了个…