HashMap详解(hashmap lru)
zhezhongyun 2025-07-21 19:06 6 浏览
讲解步骤
- 基础知识
- 工作原理
- 关键代码
- 核心方法
基础知识
数组结构
数组接口,在查询数据方面,具备优势
链表结构
链表结构,在增删数据方面,具备优势
红黑树结构
红黑树结构,在查询数据方面,数据量较大的时候,具备一定的优势
什么是散列(哈希)表
散列表,顾名思义,就是将数据分布在不同的列
但是散列表并不是完全将数据分散在不同的列,而是按照某种规则,将具备同样规则的数据存储在同一列。
即具备相同规则的数据存储在同一列,规则不同的数据分布在不同的列。
这种规则最终的产生与哈希值有关。
这里需要注意的事,哈希值只是确定最后存储列的因素,也就是说不同的哈希值可能会存在同一列。
什么是哈希值
哈希值简单的说,就是hashCode方法产生的值。
默认的hashCode方法是由其地址值最终产生一个哈希值。
由于HashMap中的元素是否存储是由键来决定,所以如果自定义的类需要存储在键,且想遵循自己的存储规则,需要重写HashCode方法
又因为Map集合的键是不能重复的,所以需要重写equals方法,定义去重规则。
工作原理
存储结构
HashMap基于散列法,又称哈希法:数组+链表+红黑树。
HashMap需要同时存储一对键和值。
Map集合中提供了put(key, value)方法,所有的键和值会被封装到一个Entry实现类(Node)对象,存储到集合中。
在存储的过程中,会先通过hashCode()方法获取一个哈希值,并通过这个哈希值,与数组的长度进行一定的运算,得到一个索引值(存储的列)
在通过equals方法来判断这个元素是否已存在,不存在则存储在该列,若存储,则保留原来的数据。
存储在一列的数据,将以链表的形式,前后关联,这样有利于将来进行删除的时候提高效率。
但是如果一列的桶结构数据过多,就会导致查询的效率降低。
为了优化桶结构带来的问题,HashMap中会去检查,当一列的桶结构数据达到8个以上,就降这一列树化(转变为树结构)
名词理解
所有的数据都是以Node节点为单位。
hash值:哈希值,该方法内部提供了一个扰动函数------int hashCode()
扰动函数:用于产生哈希值,前16位与后16位做异或运算,提高低位随机性。------h = key.hashCode()) ^ (h >>> 16)
路由寻址:由数组长度与哈希值产进行与操作,产生最终的存储列(索引位置):(table.length-1)&node.hash
Hash碰撞:哈希值如果相同,就会存储到相同的列。
链化:哈希值相同,就会存储在同系列,产生桶状结构,桶结构过长,查询数据低效。
红黑树:jdk8引入,类似于二叉树,可以避免过长的桶状结构
扩容原理
扩容:增加数组长度。目的在于解决数据过多,链化严重,默认以两倍的长度扩容。
①一列添加第8+个元素,且数组长度小于64,会优先扩容。
②一列添加第8+个元素,且数组长度达到64个,会优先树化。
③添加元素后,若哈希表中元素总个数超过阈值(一个指定的值),会进行扩容。
④扩容后,会重新根据数组长度和哈希值计算存储位置。
关键代码
核心字段
static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; 默认数组大小
static final int MAXIMUM_CAPACITY = 1 << 30; 数组最大长度
static final float DEFAULT_LOAD_FACTOR = 0.75f; 默认负载因子
static final int TREEIFY_THRESHOLD = 8; 树化阈值
static final int UNTREEIFY_THRESHOLD = 6; 树降级阈值
static final int MIN_TREEIFY_CAPACITY = 64; 树化阈值
transient Node<K,V>[] table; 哈希表
transient Set<Map.Entry<K,V>> entrySet; 键值对对象集合
transient int size; 元素长度
transient int modCount; 增删元素次数
int threshold;扩容阈值 扩容阈值=loadFactor*capacity
final float loadFactor; 负载因子
核心方法
put-->putVal(存储数据)
final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
boolean evict) {
Node<K,V>[] tab; Node<K,V> p; int n, i;
//判断表是否为空或长度为0,若满足条件,则初始化表(体现了延迟加载)
if ((tab = table) == null || (n = tab.length) == 0)
n = (tab = resize()).length;
//判断要添加的元素对应的列是否为空,若满足条件,则直接插入
if ((p = tab[i = (n - 1) & hash]) == null)
tab[i] = newNode(hash, key, value, null);
else {
Node<K,V> e; K k;
//判断元素的哈希值与要存储列的键相同,则替换键对应的值
if (p.hash == hash &&
((k = p.key) == key || (key != null && key.equals(k))))
e = p;
else if (p instanceof TreeNode)
//如果当前节点是一个数结构节点,按照树结构存储新元素。
e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
else {
for (int binCount = 0; ; ++binCount) {
//遍历当前列的节点,判断如果当前节点超过8个节点,则将当前列转为树结构。
if ((e = p.next) == null) {
p.next = newNode(hash, key, value, null);
if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
treeifyBin(tab, hash);
break;
}
if (e.hash == hash &&
((k = e.key) == key || (key != null && key.equals(k))))
break;
p = e;
}
}
//存在相同键,就值替换新值
if (e != null) { // existing mapping for key
V oldValue = e.value;
if (!onlyIfAbsent || oldValue == null)
e.value = value;
afterNodeAccess(e);
return oldValue;
}
}
//记录操作次数
++modCount;
//判断元素个数达到指定的阈值,则进行扩容操作。
if (++size > threshold)
resize();
afterNodeInsertion(evict);
return null;
}
resize(扩容)
final Node<K,V>[] resize() {
Node<K,V>[] oldTab = table;
int oldCap = (oldTab == null) ? 0 : oldTab.length;
int oldThr = threshold;
int newCap, newThr = 0;
if (oldCap > 0) {
if (oldCap >= MAXIMUM_CAPACITY) {
threshold = Integer.MAX_VALUE;
return oldTab;
}
else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&
oldCap >= DEFAULT_INITIAL_CAPACITY)
//修改新表的长度为旧表的两倍
newThr = oldThr << 1; // double threshold
}
else if (oldThr > 0) // initial capacity was placed in threshold
newCap = oldThr;
else { // zero initial threshold signifies using defaults
newCap = DEFAULT_INITIAL_CAPACITY;
newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);
}
if (newThr == 0) {
float ft = (float)newCap * loadFactor;
newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?
(int)ft : Integer.MAX_VALUE);
}
threshold = newThr;
@SuppressWarnings({"rawtypes","unchecked"})
Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap];
table = newTab;
//将新表内容,重新计算位置后,放入新表
if (oldTab != null) {
for (int j = 0; j < oldCap; ++j) {
Node<K,V> e;
if ((e = oldTab[j]) != null) {
oldTab[j] = null;
if (e.next == null)
newTab[e.hash & (newCap - 1)] = e;
else if (e instanceof TreeNode)
((TreeNode<K,V>)e).split(this, newTab, j, oldCap);
else { // preserve order
Node<K,V> loHead = null, loTail = null;
Node<K,V> hiHead = null, hiTail = null;
Node<K,V> next;
do {
next = e.next;
if ((e.hash & oldCap) == 0) {
if (loTail == null)
loHead = e;
else
loTail.next = e;
loTail = e;
}
else {
if (hiTail == null)
hiHead = e;
else
hiTail.next = e;
hiTail = e;
}
} while ((e = next) != null);
if (loTail != null) {
loTail.next = null;
newTab[j] = loHead;
}
if (hiTail != null) {
hiTail.next = null;
newTab[j + oldCap] = hiHead;
}
}
}
}
}
return newTab;
}
tableSizeFor(数组长度初始化)
二进制位运算
右移:二进制数据向右移动一位,最高位补原最高位值,原最低位舍弃。4>>1结果等于2 2>>1结果等于1
无符号右移:二进制数据向右移动一位,最高位补0,原最低位舍弃。4>>>1结果等于2 2>>>1结果等于1
无符号右移动,会确保移动后一定是一个正数。
左移:二进制数据向左移动一位,最低位补0,原最高位舍弃。举例:4<<1结果等于8 8<<1结果等于16
或:有1则1 1001|100结果为1100(12)
static final int tableSizeFor(int cap) {
//下列操作的最终目的保证了,最终的n值一定比cap大,且最接近满足+1后数组长度定义的数值(0,3,7,15,31,63...)
1001 100
int n = cap - 1;
n |= n >>> 1;
n |= n >>> 2;
n |= n >>> 4;
n |= n >>> 8;
n |= n >>> 16;
return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
}
- 上一篇:升级内核,UC浏览器电脑版v2.0.570.0下载
- 已经是最后一篇了
相关推荐
- HashMap详解(hashmap lru)
-
讲解步骤基础知识工作原理关键代码核心方法基础知识数组结构数组接口,在查询数据方面,具备优势链表结构链表结构,在增删数据方面,具备优势红黑树结构红黑树结构,在查询数据方面,数据量较大的时候,具备一定的优...
- 升级内核,UC浏览器电脑版v2.0.570.0下载
-
IT之家(www.ithome.com):升级内核,UC浏览器电脑版v2.0.570.0下载7月21日晚,UC浏览器电脑版发布了v2.0正式版,最新的版本号是2.0.570.0。此次更新的最大亮点是内...
- MySQL惊天陷阱:left join时选on还是where?
-
前天写SQL时本想通过AleftBjoinonand后面的条件来使查出的两条记录变成一条,奈何发现还是有两条。后来发现joinonand不会过滤结果记录条数,只会根据and后的条...
- 一个高效使用cursor开发项目的方法,怎么让 AI 写的代码不跑题?
-
最近又用cursor做了一个小应用,番茄时钟,用来管理自己的时间,提高效率。然后使用cursor开发的过程中。有了一些新的感悟。找到了一条可以让Curosr不跑题的办法。生成一份详细的项目资...
- 某通信公司笔试题,你会做几道?(通信行业行测题)
-
笔试部分1.描述下面代码中两个static各自的含义:staticvoidfunc(void){staticunsignedinti;}参考答案:行1,static表示静态...
- 新手学Python避坑,学习效率狂飙! 二、Python 代码缩进
-
在Python里,缩进有着极其重要的作用,它被用于界定代码块。与其他多数编程语言使用大括号{}来划分代码块不同,Python依靠缩进来明确代码的层次结构。下面分享缩进问题,并且给出避坑的方法...
- Python缩进规范详解与最佳实践(python缩进讲解)
-
Python的缩进规则是强制性的语法要求,绝对不能乱缩进!作为一门用缩进表示代码块的语言,Python的缩进错误会导致程序直接崩溃。以下是关于缩进规范的全面解析:一、Python缩进的底层逻辑代...
- 深度解析ConcurrentHashMap1.8源码分析
-
想必大家对HashMap数据结构并不陌生,JDK1.7采用的是数组+链表的方式,JDK1.8采用的是数组+链表+红黑树的方式。虽然JDK1.8对于HashMap有了很大的改进,提高了存取效率,但是线程...
- 星河战神暴风旋转刃好不好 暴风旋转刃属性详解
-
星河战神暴风旋转刃好不好?暴风旋转刃怎么得?小编为大家分享星河战神暴风旋转刃属性图鉴,希望可以帮助到大家。暴风旋转刃85级满级属性战力加成暴风旋转刃分类:僚机初始:1星属性:风满星:5星暴风旋...
- 假面骑士亚极陀全形态盘点(假面骑士亚极陀形态合集)
-
《假面骑士亚极陀》的形态设计延续了空我的属性差异化思路,但更强调“觉醒”与“进化”的主题。从基础形态的属性专精到闪耀形态的等离子爆发,每个形态都承载着角色成长与剧情转折。数据对比不仅揭示了力量...
- 膛线磨没了子弹精度到底是怎么受影响的呢?
-
这是一个回复:那么咱们就来深究这个问题吧——当膛线磨没了,子弹的精度到底是怎么受影响的呢?到底是不是就成了“滑膛枪”射程、精度是不是差了一点?先说结论——膛线磨没了,只能让这把枪成为“信仰之枪”,根本...
- 【莫莫老师亲授】公考行测图形推理3大属性规律!提分利器
-
在公务员考试的行测科目中,图形推理是高频考点之一,而属性规律作为基础题型中的核心模块,常通过图形的对称性、曲直性、开闭性三大属性考查考生的观察与归纳能力。本文将从定义、特点、常见考法及例题分析入手,系...
- DNF驱魔属性合并,技能重做解读(dnf驱魔加强了吗)
-
作者:B蜀黍前言本轮职业平衡驱魔进行了力驱法驱合二为一,部分技能重做和技能优化。形态方面:新增四神系技能且可柔化,力系技能加快出手速度,一觉从1秒瞬发单段脱手改为3秒站桩多段,形态逆向优化,实战体验...
- 拉满一个3级配件?还是升6个2级配件,橙装升级最优解方案揭秘!
-
各位车长,试验型配件已经重新发放!即日起至3月31日,在这期间登录游戏就能免费领到6个试验型配件和700个合金原件(已领取的玩家不再重复获得)。今天就和各位车长简单聊一聊:活动期间一共能获得多少合金元...
- CAD动态块制作方法—旋转动态块(cad动态块制作教程 视频教程)
-
CAD动态块制作方法—可见性动态块CAD动态块制作方法—拉伸动态块今天来讲一下含有旋转动作的动态块制作方法。我们用立面索引符号来举例。旋转动态块首先制作一个带有属性定义的立面索引符号的图块。(此步...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 教程 (33)
- HTML 简介 (35)
- HTML 实例/测验 (32)
- HTML 测验 (32)
- JavaScript 和 HTML DOM 参考手册 (32)
- HTML 拓展阅读 (30)
- HTML文本框样式 (31)
- HTML滚动条样式 (34)
- HTML5 浏览器支持 (33)
- HTML5 新元素 (33)
- HTML5 WebSocket (30)
- HTML5 代码规范 (32)
- HTML5 标签 (717)
- HTML5 标签 (已废弃) (75)
- HTML5电子书 (32)
- HTML5开发工具 (34)
- HTML5小游戏源码 (34)
- HTML5模板下载 (30)
- HTTP 状态消息 (33)
- HTTP 方法:GET 对比 POST (33)
- 键盘快捷键 (35)
- 标签 (226)
- HTML button formtarget 属性 (30)
- CSS 水平对齐 (Horizontal Align) (30)
- opacity 属性 (32)