HashMap结构

数组+链表+红黑树
链表大于8转红黑树,红黑树节点数小于6退回链表。

存放的key-value的Node节点

static class Node implements Map.Entry {        final int hash;        final K key;        V value;        Node next;    }

树形结构的Node节点

 static final class TreeNode extends LinkedHashMap.Entry {        TreeNode parent;  // red-black tree links        TreeNode left;        TreeNode right;        TreeNode prev;    // needed to unlink next upon deletion        boolean red; }

他的继承结构是这样,可以看到继承了Node节点

看懂一条语句

 hash & tab.length-1

代码中多处都可以看到这条代码,实际上这条语句只是做了一个取余(%)的动作。一个&怎么做的取余的操作:HashMap的容量为2^n其二进制结构如下

任何数&2^n-1(01111…)其结果都是去0xxxx,做了快速取余的操作。后续会看到该条语句频繁出现

几个核心参数

  • Node[] table:存放Node的数组
  • int size:表示当前HashMap包含的键值对数量
  • int threshold:表示当前HashMap能够承受的最多的键值对数量,一旦超过这个数量HashMap就会进行扩容
  • final float loadFactor:负载因子,用于扩容
  • int DEFAULT_INITIAL_CAPACITY = 16:默认的table初始容量
  • float DEFAULT_LOAD_FACTOR = 0.75f:默认的负载因子
  • int TREEIFY_THRESHOLD = 8: 链表长度大于该参数转红黑树
  • int UNTREEIFY_THRESHOLD = 6: 当树的节点数小于等于该参数转成链表

初始化方法

指定了具体的容量,以及负载因子的初始化方法。当知道需要放入的元素的个数时可以先指定避免多次扩容造成性能浪费。

public HashMap(int initialCapacity) {        this(initialCapacity, DEFAULT_LOAD_FACTOR);}  public HashMap(int initialCapacity, float loadFactor) {        this.loadFactor = loadFactor;        this.threshold = tableSizeFor(initialCapacity);}

核心方法 public V get(Object key)

参数key,以及该key的hash
先判断数组是否已经初始化了,以及数组长度。
在判断tab[(n - 1) & hash],前文提到的那一条语句,用key的hash取余数组长度判断数组中的位置是否存在元素。

  • 不存在元素
    数组中不存在元素,肯定没有产生hash冲突,那么元素肯定不存在
  • 数组当前位置存在元素
    判断key的hsah相等并且(key的地址相等或者equals相等)。那么可以确定元素在数组中。
  • 数组当前位置存在元素,但是key不相等
    接下来会去判断数组中当前位置是否存在next元素(Node节点结构),如果有next说明存在链表或者树形结构。
    接下来判断Node是否是TreeNode,如果是则按照遍历树方式遍历得到结果,不是则按照遍历链表的形式遍历得到结果。
final Node getNode(int hash, Object key) {        Node[] tab = table;         Node first = tab[(n - 1) & hash];         Node e = first.next;         int n = tab.length;         K k;        //数组是否已经初始化        if (tab!= null && n > 0 && first != null) {            //table中是否有节点,key是否相等            if (first.hash == hash && ((k = first.key) == key || (key != null && key.equals(k))))                return first;            //key不相等,判断是否有next,并且判断是树的节点还是链表的节点,再以不同的方式去遍历获取            if (e != null) {                if (first instanceof TreeNode)                    return ((TreeNode)first).getTreeNode(hash, key);                do {                    if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k))))                        return e;                } while ((e = e.next) != null);            }        }        return null;    }

核心方法 public V put(K key, V value)

put方法比较长,分几种情况解析

  • 第一次put元素,数组还未初始化:调用resize()初始化数组,直接放入table相应位置。(resize扩容方法很重要)
  • table中该位置没产生Hash冲突:构造节点放入table中
  • 产生Hash冲突,先判断table中节点元素key是否相等,相等则替换value
  • 产生Hash冲突,节点是TreeNode:按红黑树的方式插入节点
  • 产生Hash冲突,节点是Node:构造节点按链表的方式插入,并且检查插入后是否到达转红黑树的阈值8
final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) {        Node[] tab;         Node p;         int n;        int i;        //第一次put元素,数组还未初始化        if ((tab = table) == null || (n = tab.length) == 0)            //调用resize()初始化数组            n = (tab = resize()).length;         //table中该位置没产生Hash冲突        if ((p = tab[i = (n - 1) & hash]) == null)            //构造节点放入table中            tab[i] = newNode(hash, key, value, null);        else {            //产生Hash冲突            Node e; K k;            //table中节点元素key是否相等            if (p.hash == hash && ((k = p.key) == key || (key != null && key.equals(k))))                //记录下节点的引用,后续替换E节点的值                e = p;            else if (p instanceof TreeNode)                //节点是TreeNode:按红黑树的方式插入节点                e = ((TreeNode)p).putTreeVal(this, tab, hash, key, value);            else {                //节点是Node,链表情况                for (int binCount = 0; ; ++binCount) {                    //遍历到链表尾部还未发现相同的key,则构造节点插入到链表                    if ((e = p.next) == null) {                        p.next = newNode(hash, key, value, null);                        if (binCount >= TREEIFY_THRESHOLD - 1)                            //检查插入后是否到达转红黑树的阈值8                            treeifyBin(tab, hash);                        break;                    }                    //检查是否有相同的key,有就退出,e = p.next已经记录了E的引用                    if (e.hash == hash &&                        ((k = e.key) == key || (key != null && key.equals(k))))                        break;                    p = e;                }            }            //上述各种情况下,如果不是插入节点的情况下            //存在key相同的情况下,完成一个值的替换            if (e != null) {                V oldValue = e.value;                if (!onlyIfAbsent || oldValue == null)                    e.value = value;                return oldValue;            }        }        //检查是否需要扩容 存在的数量>hashmap容量*负载因子就需要扩容        if (++size > threshold)            resize();        return null;    }

核心方法Node[] resize()

最重要的部分来了,也是面试官最喜欢问的HashMap的扩容

  1. 第一次初始化时候调用
  2. 存放的键值对的数量>hashmap容量*负载因子就需要扩容

初始化时候调用resize

扩容后得到的是一个Node数组。由于第一次初始化,肯定是不存在链表,红黑树等结构的,以及Node节点的。只是对一些属性做了赋值操作,和返回一个空的Node数组。threshold:HashMap能够承受的最多的键值对数量;如果指定了容量和负载因子,则threshold = 指定的容量*负载因子;Node[] table:存放Node的数组,创建了一个容量为16(未指定具体容量时,默认为16)的Node数组。

省略部分与第一次初始化无关代码不重要的代码

final HashMap.Node[] resize() {        //16*0.75=12        threshold = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);        //16        HashMap.Node[] newTab = (HashMap.Node[])new HashMap.Node[DEFAULT_INITIAL_CAPACITY];        table = newTab;        return newTab;    }

扩容调用resize 重点来了

先删除一些与扩容无关的代码

final Node[] resize() {        Node[] oldTab = table;        int oldCap = (oldTab == null) ? 0 : oldTab.length;        int oldThr = threshold;        int newCap, newThr = 0;        if (oldCap > 0) {            //容量和阈值都扩大成两倍            newCap = oldCap << 1;            newThr = oldThr << 1;        }        //设置阈值属性        threshold = newThr;        //新建一个是之前两倍容量的大小的Node数组        Node[] newTab = (Node[])new Node[newCap];        //属性赋值        table = newTab;        //完成一些准备,开始准备迁移之前节点        if (oldTab != null) {            //循环迁移每个节点数据            for (int j = 0; j < oldCap; ++j) {                Node e;                //并不数组中是每个位置都有元素                if ((e = oldTab[j]) != null) {                    //数据需要迁移,table相应位置置空                    oldTab[j] = null;                    //不存在链表情况                    if (e.next == null)                        //重新对新的容量快速取余,放入相应的位置                        newTab[e.hash & (newCap - 1)] = e;                    //树结构(较为复杂后续再分析)                    else if (e instanceof TreeNode)                        ((TreeNode)e).split(this, newTab, j, oldCap);                    //链表结构                    else {                        //不需要移动的链表的头尾指针                        Node loHead = null, loTail = null;                        //需要移动的链表的头尾指针                        Node hiHead = null, hiTail = null;                        Node 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);                        //将链表重新放入table数组中                        if (loTail != null) {                            loTail.next = null;                            newTab[j] = loHead;                        }                        if (hiTail != null) {                            hiTail.next = null;                            newTab[j + oldCap] = hiHead;                        }                    }                }            }        }        return newTab;    }

这里主要分析一下拆分的依据。
当要将链表中的数据进行拆分,并且分配到不同的table下标中。可以明确的是,不能因为扩容影响到get方法,所以根据get方法key的hash取余容量可以得到如下两张图片。未拆之前

拆之后

扩容中读懂一行代码

(e.hash & oldCap) == 0

当元素的hash &oldCap(前文提到过容量为2^n,其二进制为1000…)。
看几个例子
5&16 ==0

21&16 !=0

69&16 ==0

Hash值的结构以红色框为中心:可以看成 左边Y+table容量+右边X,Y是容量的偶数倍数,X小于容量值,从上述例子中可以看出来结果是否为0取决于红色框处是0还是1。

  • 为0结果恒为0
    Y是容量的偶数倍数扩容后取余为依旧0,余数为X,余数与扩容之前一致,不需要移动。
  • 为1时结果不为0
    Y是容量的偶数倍数扩容后取余为依旧0,余数为容量值+X,扩容后余数与之前不一致,需要移动,移动后的位置为容量+X(之前所在位置的值)。

最后

欢迎关注公众号:前程有光,领取一线大厂Java面试题总结+各知识点学习思维导+一份300页pdf文档的Java核心知识点总结!

hashmap转红黑树的阈值为8_面试必问的HashMap,一次彻底帮你搞定HashMap源码相关推荐

  1. hashmap转红黑树的阈值为8_面试必考的 HashMap,这篇总结到位了

    点击蓝色"JavaKeeper"关注我哟加个"星标",一起成长,做牛逼闪闪的技术人 1 概述 HashMap是基于哈希表实现的,每一个元素是一个key-valu ...

  2. hashmap中用红黑树不用其他树_HashMap面试专题:常问六题深入解析

    引言 其实我很早以前就想写一篇关于HashMap的面试专题.对于JAVA求职者来说,HashMap可谓是集合类的重中之重,甚至你在复习的时候,其他集合类都不用看,专攻HashMap即可. 然而,鉴于网 ...

  3. 一招彻底帮你搞定HashMap源码,项目实战

    ES 集群架构演进之路 1.初始阶段 订单中心ES初始阶段如一张白纸,架设方案基本没有,很多配置都是保持集群默认配置.整个集群部署在集团的弹性云上,ES集群的节点以及机器部署都比较混乱.同时按照集群维 ...

  4. 一招彻底帮你搞定HashMap源码,成长路线图

    kafka面试基础[17] 1.Kafka的用途有哪些?使用场景如何? 2.Kafka中的ISR.AR又代表什么?ISR的伸缩又指什么 3.Kafka中的HW.LEO.LSO.LW等分别代表什么? 4 ...

  5. 关于hash,hashCode, hashMap,红黑树

    小刘老师讲HashMap源码 小刘老师讲红黑树 hash,hashCode,hashMap 1.前提知识 数组: 列表: 散列表 什么是hash? hash也称散列,哈希,基本原理就是把任意长度的输入 ...

  6. 面试系列Java中级:为什么HashMap引入红黑树?

    因为在JDK1.7之前,HashMap采用数组+链表实现,即使用链表处理冲突,同一hash值的链表都存储在一个链表里.但是当位于一个数组中的元素较多,即hash值相等的元素较多时,通过key值依次查找 ...

  7. 【HashMap原理+红黑树】

                    HashMap原理+红黑树 一.HashMap原理 HashMap采用Entry数组来存储key-value对,每一个键值对组成了一个Entry实体,Entry类实际上 ...

  8. Android技术栈(五)HashMap(包括红黑树)与ArrayMap源码解析

    1 总览 本文会对 Android 中常用HashMap(有红黑树)和ArrayMap进行源码解析,其中 HashMap 源码来自 Android Framework API 28 (JDK=1.8) ...

  9. hashmap中用红黑树不用其他树_为什么hashMap引入了红黑树而不是其他结构

    1.为什么hashMap使用红黑树而不是其他结构? 在回答这个问题之前,我们先了解一下有关二叉树的基本内容. ①二叉排序树(又称二叉查找树): 1)若左子树不为空,则左子树上所有结点的值均小于根结点的 ...

最新文章

  1. MySQL 数据库常用命令
  2. 贝叶斯分类器的matlab实现_贝叶斯实验
  3. opencv函数findcontours_OpenCV 中的轮廓应用
  4. uln2003驱动蜂鸣器_让蜂鸣器发声
  5. django-视图函数 00
  6. Android逆向系列之ARM语法篇
  7. 【渝粤教育】电大中专工程图学基础 (2)作业 题库
  8. case when 效率_采用机械涡轮复合增压系统优化7.8 L柴油机的 稳态效率和排放性能...
  9. java实习生面试一些技巧
  10. 阿里一p7员工为了证明自己确实年入百万,晒出了他的工资
  11. App应用最有效的变现方式,还能同时提升留存!
  12. 这辈子你会遇见谁,早已命中注定
  13. 沐风:了不起的便利店
  14. m1/m1Pro/m1Max芯片下载win11-arm镜像
  15. Java中String 字符串与List<String>互转
  16. python截取视频_python+ffmpeg截取视频段
  17. 模块 , 用户管理系统 , 购物车程序 , 分页显示.
  18. 网上花店java项目_Java+SSM实现网上花店售卖系统
  19. 收藏一个白嫖资源的网站链接
  20. 使用多个关键字对word进行批量查找统计

热门文章

  1. js输出php文件大小,前端js实现文件的断点续传 后端PHP文件接收
  2. bos 获取数据库连接_java解析数据接口获取json对象
  3. android 点击网络图片大全,android查看网络图片的实现方法
  4. mysql的repeat()函数
  5. 被final修饰的变量在哪存储_final,static,this,super 关键字总结,一点课堂(多岸学院)...
  6. go语言学习,channel消费者和生产者
  7. 解决MySQL8.0报错:Unknown system variable 'validate_password_policy'
  8. matlab mnl,LaTex中插入VISO 和 MATLAB 经验总结
  9. 松下服务器分频器输出信号与,基础资料松下PANASONIC伺服驱动器MADHT1507E
  10. java word 纸张大小_如何在Java中为Word文档(.doc或.docx)设置背景色(页面颜色)?...