讲讲HashMap?

源码解析

final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
boolean evict) {
//辅助变量
Node<K,V>[] tab; Node<K,V> p; int n, i;
//如果当前tabe数组是null,数量是0的话
if ((tab = table) == null || (n = tab.length) == 0)
//执行扩容方法resize()
//初始化数组默认大小是16
n = (tab = resize()).length;
//通过hash与运算获得数组索引位置,该位置是null
if ((p = tab[i = (n - 1) & hash]) == null)
//创建新Node对象
tab[i] = newNode(hash, key, value, null);
else {//该索引位置不为null
Node<K,V> e; K k;
//当前table索引hash和新的索引hash相同 &&
//(当前table节点的key和新key是同一对象 || equals为真)
if (p.hash == hash &&
((k = p.key) == key || (key != null && key.equals(k))))
e = p;
//如果当前table的node已是红黑树就按红黑树处理
else if (p instanceof TreeNode)
e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
else {
//如果节点是后面是链表就遍历比较
for (int binCount = 0; ; ++binCount) {
//遍历整个链表结束,也没有和新的key相同的就添加链表后面新创建node
if ((e = p.next) == null) {
p.next = newNode(hash, key, value, null);
//加入后判断当前的链表个数是否已经达到8个,如果达到就进行红黑树转换
if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
//提示:
//treeifyBin里如果table为null或者大小小于64,暂时不会转化为红黑树
//而是进行扩容。
//if (tab == null || (n = tab.length) < MIN_TREEIFY_CAPACITY)
treeifyBin(tab, hash);
break;
}
//发现相同就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;
//每增加node,Size++
//如果数量大于临界值,就进行扩容
//threshold=table大小(默认16)*负载因子(默认0.75)
//12>24>>48
if (++size > threshold)
//扩容成2倍增长16->32->64
resize();
afterNodeInsertion(evict);
return null;
}

简述

Jdk1.7 数组+链表

Jdk1.8 数组+链表+红黑树

hashMap 默认数组大小16,负载因子 默认0.75, 临界值= 数组大小*负载因子。

  1. 首先将key进行hash算法,key的hashCode右移16位并进行异或运算。
  2. 如果table是null ,先初始化数组默认大小是16
  3. 通过hash的与运算获得当前table索引位置,如果索引位置内容是null则创建新Node节点对象
  4. 如果当前table索引位置里内容不为null,则寻找相同key直接修改值,分三种情况
    1. 当前table索引的对象hash是否与新hash相同并且当前table位置对象key是否和新key相同
    2. 如果当前table索引的对象已是TreeNode(红黑树)就进行遍历树方式查找
    3. 如果是链表则遍历链表查找,如果遍历完也没找到就往链表尾部加入并创建新Node,在判断当前大小是否大于等于8,如果大于就进行红黑树转化。执行红黑树转化方法时有个条件,如果数组大小小于64则还是进行扩容操作。
  5. 当前Size数组大小和threshold(临界值)比较,如果size大于threshold则进行扩容(2倍增长),

问题

JDK 1.8中对hash算法和寻址算法是如何优化的?

//hash算法优化
static final int hash(Object key) {
int h;
return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}
//(n - 1) & hash 寻址算法优化
if ((p = tab[i = (n - 1) & hash]) == null)

配合来说,put键值时,肯定涉及数组长度的取模运算,但是在计算机上面(n - 1) & hash位运算效率高于普通的除法取余。关键这个n(hashmap长度)通常会很小,但是hash值是32位的,因此(n-1)大概率高位补零,由于与运算是只有两个数字都是1结果才为1,其他情况均为0,因此hash值的高16位永远起不到作用,这样就大大的增加了寻址冲突概率,除非数组长度很大,撑满整个32位最好都为1,才不会有冲突,但一般不可能很长,所以通过hash算法的优化(h = key.hashCode()) ^ (h >>> 16) 这个公式完美的让hash值高16位与低16位融合,保留高低16位的特征。再跟(n-1)做与运算,hash冲突的概率就降低了!

深度解析HashMap的更多相关文章

  1. 深度解析HashMap底层实现架构

    摘要:分析Map接口的详细使用以及HashMap的底层是如何实现的? 本文分享自华为云社区<[图文并茂]深度解析HashMap高频面试及底层实现结构![奔跑吧!JAVA]>,原文作者:灰小 ...

  2. 深度解析HashMap集合底层原理

    目录 前置知识 ==和equals的区别 为什么要重写equals和HashCode 时间复杂度 (不带符号右移) >>> ^异或运算 &(与运算) 位移操作:1<&l ...

  3. 源码深度解析SpringMvc请求运行机制(转)

    源码深度解析SpringMvc请求运行机制 本文依赖的是springmvc4.0.5.RELEASE,通过源码深度解析了解springMvc的请求运行机制.通过源码我们可以知道从客户端发送一个URL请 ...

  4. mybatis 3.x源码深度解析与最佳实践(最完整原创)

    mybatis 3.x源码深度解析与最佳实践 1 环境准备 1.1 mybatis介绍以及框架源码的学习目标 1.2 本系列源码解析的方式 1.3 环境搭建 1.4 从Hello World开始 2 ...

  5. Flink 源码解析 —— 深度解析 Flink 是如何管理好内存的?

    前言 如今,许多用于分析大型数据集的开源系统都是用 Java 或者是基于 JVM 的编程语言实现的.最着名的例子是 Apache Hadoop,还有较新的框架,如 Apache Spark.Apach ...

  6. Spring源码深度解析之Spring MVC

    Spring源码深度解析之Spring MVC Spring框架提供了构建Web应用程序的全功能MVC模块.通过策略接口,Spring框架是高度可配置的,而且支持多种视图技术,例如JavaServer ...

  7. [WebKit内核] JavaScript引擎深度解析--基础篇(一)字节码生成及语法树的构建详情分析

    [WebKit内核] JavaScript引擎深度解析--基础篇(一)字节码生成及语法树的构建详情分析 标签: webkit内核JavaScriptCore 2015-03-26 23:26 2285 ...

  8. 第37课 深度解析QMap与QHash

    1. QMap深度解析 (1)QMap是一个以升序键顺序存储键值对的数据结构 ①QMap原型为 class QMap<K, T>模板 ②QMap中的键值对根据Key进行了排序 ③QMap中 ...

  9. Deep Learning模型之:CNN卷积神经网络(一)深度解析CNN

    http://m.blog.csdn.net/blog/wu010555688/24487301 本文整理了网上几位大牛的博客,详细地讲解了CNN的基础结构与核心思想,欢迎交流. [1]Deep le ...

随机推荐

  1. pytest-rerunfailures/pytest-repeat重跑插件

    在测试中,我们会经常遇到这种情况,由于环境等一些原因,一条case运行5次,只有两次成功 其它三次失败,针对这种概率性成功或失败,若是我们每次都运行一次就比较耗时间,这个时候 就需要pytest提供的 ...

  2. 9.2 k8s结合Jenkins与gitlab实现代码升级与回滚

    1.部署Jenkins 1.1 安装jdk # apt 安装jdk11 apt install openjdk-11-jdk # 查看 root@jenkins:~# java -version op ...

  3. File与IO基础

    IO流的作用:持久化到磁盘 File类的使用 File类基本概念 文件和文件夹都是用File类来表示. File类是内存层面的对象,内存中创建出来的File对象不一定有一个真实存在的文件或文件夹,但是 ...

  4. CF1575G GCD Festival

    \(\sum\sum gcd(i,j) \times gcd(a_i,a_j)\) 考虑枚举这个 \(gcd(i,j)\) . \(\sum_d \varphi(d)\sum_{i|d}\sum_{j ...

  5. Codeforces 605D - Board Game(树状数组套 set)

    Codeforces 题目传送门 & 洛谷题目传送门 事实上是一道非常容易的题 很容易想到如果 \(c_i\geq a_j\) 且 \(d_i\geq b_j\) 就连一条 \(i\to j\ ...

  6. Codeforces 453E - Little Pony and Lord Tirek(二维线段树+ODT)

    Codeforces 题目传送门 & 洛谷题目传送门 一道难度 *3100 的 DS,而且被我自己搞出来了! 不过我终究还是技不如人,因为这是一个 \(n\log^2n\) + 大常数的辣鸡做 ...

  7. 【机器学习与R语言】8- 神经网络

    目录 1.理解神经网络 1)基本概念 2)激活函数 3)网络拓扑 4)训练算法 2.神经网络应用示例 1)收集数据 2)探索和准备数据 3)训练数据 4)评估模型 5)提高性能 1.理解神经网络 1) ...

  8. 一个画组织解剖图R包

    地址: https://github.com/jespermaag/gganatogram

  9. 安卓手机添加系统证书方法(HTTPS抓包)

    目录 1. 导出证书(以Charles为例) 2. 安卓证书储存格式 3. 将导出的证书计算hash值 4. 生成系统系统预设格式证书文件 5. 上传证书 安卓7.0以后,安卓不信任用户安装的证书,所 ...

  10. 求最长子序列(非连续)的STL方法 - 洛谷P1020 [NOIP1999 普及组] 导弹拦截

    先给出例题:P1020 [NOIP1999 普及组] 导弹拦截 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 大佬题解:P1020 [NOIP1999 普及组] 导弹拦截 - 洛谷 ...