hashmap扩容 面试_HashMap面试,看完这一篇就够了(上)

以下HashMap源码的解析都是基于java8来讲解的。

HashMap的结构是数组加链表的形式(jdk7中也是),在java8中引入了红黑树,由于红黑树的时间复杂度是O(log n),引入红黑树是为了解决在哈希冲突很严重的时候导致链表太长,从而引起的查找效率太低的问题。

常量

※ 为什么链表长度大于8才进行树化

源码中有下面一段解释

大体意思就是:由于TreeNode所占空间是普通Node的两倍,所以只有在bin包含足够多的node的时候才会转化为Tree(有TREEIFY_THRESHOLD决定)。当bin变的很小的时候,又会转为链表。当hashCode分布良好的时候,几乎用不到tree,hash冲突很小。但是在随机hashCode时,离散性差,会导致hash冲突严重,所以导致链表很长,这时候就要转化为红黑树来提高查询效率。按上面java官方给出的概率,链表长度达到8的概率是0.00000006,是很低很低的概率了,所以java也是通过大概率统计得出大于8的时候才转化为红黑树。

内部类 Node

从java8开始,HashMap的节点改为了使用Node(java7使用Entry),其实都差不多,内部结构都类似。

hash hash值key key值value value值next 下一个节点方法

put方法

put方法调用了下面的putVal方法,这里和java7版本的不同有两处:

当链表长度大于8的时候,会转化为红黑树;而且如果Node是TreeNode类型,则按照红黑树的方式进行putput采用的是尾插法,这样在扩容的时候避免了多线程情况下出现死循环(java7采用头插法,会存在扩容时出现死循环)

get方法

get方法调用了getNode方法

HashMap扩容

HashMap的扩容存在的问题也是面试中经常问到的,首先来看下扩容的源码

在这段代码中,java8进行了优化的,一部分是元素位置不变的,一部分是元素位置+old capacity。它是这样实现的:

e.hash & oldCap

这段代码不是为了判断元素所在的位置,而是判断hash值在old capacity的那一位是不是0,举个例子:hashCode是20,old capacity是16,new capacity是32,计算元素位置的方式:(n - 1) & hash

扩容前后对比发现,差距就在old capacity二进制1的位置那,所以e.hash & oldCap可以判断出那个位置是否为1。如果为1,那么元素的新位置就是old capacity+扩容前的元素位置 即为扩容后的位置;如果为0,则扩容后的位置与扩容前相同。java8之所以这样优化,首先是因为HashMap的capacity始终是2的幂次方,这样就保证了e.hash & oldCap的正确性;其次这样优化去掉了扩容时的重新hash运算,提高了效率。

扩容存在的问题

数据覆盖看resize源码可以发现,当两个线程同时走到#1的位置时,如果线程1和线程2的key的hash值相同,那线程1赋值后让出cpu,此时线程2获得时间片,同样也进行了赋值操作,那么线程1赋的值就被线程2给覆盖掉了。死循环在java8中已经不存在死循环了。但是在java7中是存在的。有与java7是头插法,所以在resize的时候,链表的顺序会反转,此时两个线程同时进行resize,就有可能形成链式的圆圈,造成死循环。HashMap线程不安全,那该用什么

HashMap的线程版本有HashTable和Collections.synchronizedMap(map),但是这两个都是直接加synchronized实现的,效率很低。而HashMap的线程安全版本常用的是ConcurrentHashMap,这个也是面试中经常问到的。明天会更新ConcurrentHashMap的相关面试点,欢迎大家继续关注。

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

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

相关文章

动态更改屏幕方向LANDSCAPE与PORTRAIT 转

动态更改屏幕方向……LANDSCAPE与PORTRAITpackage irdc.ex05_22;import android.app.Activity;import android.content.pm.ActivityInfo;import android.os.Bundle;import android.view.Display;import android.view.View;import android.widget.Button;import android.widget.…

C# action,delegate,func的用法和区别

以前我都是通过定义一个delegate来写委托的,但是最近看一些外国人写的源码都是用action和func方式来写,当时感觉对这很陌生所以看起源码也觉得陌生,所以我就花费时间来学习下这两种方式,然后发现确实代码简洁了不少。这两种方式我…

java实现红包要多少钱_Java实现抢红包算法,附完整代码(公平版和手速版)

当我们在群里抢红包时真的是手速越快红包金额越大吗?答案当然是并不是,都说了是拼手气,岂能是拼手速!不过也可以有拼手速的方法二倍均值法(公平版)这是一种很合理很公平的抢红包算法了,绝对不会让你拼手速的&#xff0…

堆和栈概念整理

1.内存管理方式: 堆:一般由程序员分配释放,程序员申请时需要指明大小,leg,C语言中malloc函数:p1(char*)malloc(10),C中:p2new char[20].若程序员不去释放,当程序结束时可能由操作系统释放 &…

.Net之Swagger基础使用

介绍Swagger 是一个规范和完整的框架,用于生成、描述、调用和可视化 RESTful 风格的 Web 服务。日常可以用于后端开发人员测试接口或者前后端联调使用。从.net5开始,swagger已经集成到vs2019编译器中,可以通过勾对选项“启用OpenAPI支持”显示…

孩子不是笨,他和“最强大脑”差的是这个!

小木最近看到一则消息推送,说家长辅导孩子陪写作业简直就是一道“送命题”。朋友圈更是掀起了一股“提前嫁儿嫁女”的热潮。为什么孩子对数学一点兴趣也没有?为什么再简单的一道题,换个数字换个形式,孩子就一问三不会了&#xff1…

c++ 隐藏进程_Linux 查看进程的动态信息

前言top命令经常用来监控Linux的系统状况,是常用的性能分析工具,能够实时显示系统中各个进程的资源占用情况目录一、描述二、top命令常用字段含义三、top中的子命令四、总结五、思维导图一、描述1、top命令经常用来监控Linux的系统状况是常用的性能分析工…

给IT新人的15个建议:苦逼程序员的辛酸反省与总结

很多人表面上看着老实巴交的,实际上内心比谁都好强、自负、虚荣、甚至阴险。工作中见的多了,也就习惯了。 有一些人,什么事都写在脸上,表面上经常得罪人,甚至让人讨厌。但是他们所表现的又未必不是真性情。 我相信大多…

.net core针对async ()=的安全处理

最近在做一个功能需要传递一个委托作为回调逻辑处理,但在使用中定义了async ()>来处理awaiter逻辑那就存在一个安全问题了。了解async/awaiter的朋友一定清楚async void函数带来的致命风险!async void会阻断异常路由,即当前函数没有try的情…

这三道题,总有一道你是答不出来的

全世界只有3.14 % 的人关注了数据与算法之美大家好,我是最近过得不太舒心的卢sir。经常被小思妹提的各种奇葩数学题搞得云里雾里的。我一看数学题就想做,没想到这些题目都是奇葩中的奇葩,不是那写错就是这计算错。为了捍卫我最后的倔强&#…

前端对div连线_《前端图形学从入门到放弃》003 三维世界

从本篇起,我们将正式进入webgl的3D世界本篇涵盖的内容包括:webgl它在干啥?如何画一个正方体?如何成为一个“有深度”的正方体?正方体要离家出走了!webgl它在干啥?首先我们需要知道webgl的世界其…

为EasyUI 的Tab 标签添加右键菜单

近期研究了下MenuButton,有了新的感悟,原先在那个DEMO中右键支持做法,现在看来真是小儿科啊! 前期的准备工作: 1、下载DEMO源码,并升级为最新版本(jquery 1.7.2,Easyui 1.2.6),直接替换就可以啦…

C# Task 暂停与取消

前言:①取消task任务之CancellationTokenSource的用法;②task的线程管控方法Task..Wait(time),Task.WaitAll(), Task.WaitAny(),task.ContinueWith.1.声明参数CancellationTokenSource tokenSource new CancellationTokenSource(…

通过R,让你的数据分析更简便!

R作为一种统计分析软件,广泛应用于生物、医学、电商、新闻等数据相关行业,是目前主流数据应用软件之一。为了更好地帮助大家了解并快速入门R语言,现超级数学建模携手柯老师以R语言为基础,向大家隆重推出《R语言基础》系列课。柯老…

r语言用行名称提取数据框信息显示na_学会这些R语言技巧至少可以节省半年时间...

ubuntu备忘定期清空回收站扩增子数据牢记r ubuntu 相关技巧和备忘待解决问题1:phyloseq有一篇文章案例使用输入和输出文件相同的文件名,无法执行待解决问题2:待解决问题3:样品分组文件太长了,导致提取出来数据存在NA值…

hibernate.cfg.xml的一些事

Hibernate连接数据库的配置文件书写一般有两种方式: 第一种方式:使用开发环境直接连接数据库最后生成hibernate.cfg.xml文件 第二种方式:使用已有的模版直接拷贝的工作的目录下,通过相应的修改获得需要的连接数据库的配置文件&…

NET问答: 如何将 DataTable 转成 IEnumerableT ?

咨询区 Roddy Balkan&#xff1a;我想将一个外域系统中传过来的 DataTable 转成 IEnumerable&#xff0c;下面的代码在 ASP.NET 4.6.1 中是没有问题的。public static IEnumerable<UserAssignmentDto> StaffAssignmentsUsingStoredProcedure(System.Data.DataTable dataT…

columnproperty server sql_获取SQL Server表字段的各种属性

SELECT(CASE WHEN a.colorder1 THEN d.name ELSE END) N表名,a.colorder N字段序号,a.name N字段名,(CASE WHEN COLUMNPROPERTY( a.id,a.name,IsIdentity)1 THEN √ELSE END) N标识,(CASE WHEN (SELECT COUNT(*)FROM sysobjectsWHERE (name in(SELECT nameFROM sysindexesWHE…

配置静态路由下一跳为本地出战接口和IP地址的区别

配置静态路由下一跳为本地出战接口和IP地址的区别 在配置静态路由时&#xff0c;下一跳可以使用下一路由器的IP地址&#xff0c;也可以使用本路由器的出站接口。在点对点的网络中&#xff0c;两者可能没有什么差别&#xff0c;但在以太网中或者NBMA网络类型&#xff0c;两者有很…

使用 Source Generator 自动生成 WEB API

使用 Source Generator 自动生成 WEB APIIntro上次我们介绍了使用 Source Generator 的应用&#xff0c;有小伙伴留言说想要自动生成一套 ABP 相关的东西&#xff0c;我对 ABP 不怎么熟悉&#xff0c;所以写了一个简单版的雏形&#xff0c;可以根据自定义的模板去动态生成&…