首页 > 其他分享 >HasMap底层分析

HasMap底层分析

时间:2024-03-24 11:59:48浏览次数:20  
标签:分析 hash HashMap 链表 key 0000 null HasMap 底层

一、散列表结构

HashMap的存储结构为数组+链表+红黑树

同时它的数组的默认初始容量是 16、扩容因子为 0.75,每次采用 2 倍的扩容。也就是说,每当我们数组中的存储容量达到 75%的时候,就需要对数组容量进行 2 倍的扩容。

初始容量和负载因子也可以通过构造方法指定:

 public HashMap(int initialCapacity, float loadFactor) {
     if (initialCapacity < 0)
         throw new IllegalArgumentException("Illegal initial capacity: " +
                                            initialCapacity);
     if (initialCapacity > MAXIMUM_CAPACITY)
         initialCapacity = MAXIMUM_CAPACITY;
     if (loadFactor <= 0 || Float.isNaN(loadFactor))
         throw new IllegalArgumentException("Illegal load factor: " +
                                            loadFactor);
     this.loadFactor = loadFactor;
     this.threshold = tableSizeFor(initialCapacity);
 }

二、HashMap 的 put 过程

HaahMap 使用 put的方式进行数据的存储,其中有两个参数,分别是 keyvalue

HashMap 将将要存储的值按照 key计算其对应的数组下标,如果对应的数组下标的位置上是没有元素的,那么就将存储的元素存放上去,但是如果该位置上已经存在元素了,那么这就需要用到我们上面所说的链表存储了,将数据按照链表的存储顺序依次向下存储就可以了。存储结果如下:

链表插入:在 JDK1.7 以及前是在头结点插入的,在 JDK1.8 之后是在尾节点插入的。

当需要存储的数据很多时,为了避免链表太长,会进行树化

树化:当链表长度大于8时,会对链表进行树化操作,将其转换为一课红黑树(二叉树,左边节点的值小于根节点,右边节点的值大于根节点),这样在查找元素时类似于二分查找,提升查询效率。

当进行删除操作而使的树上的节点减少后,链表的长度不再大于8了,这时就要进行链化。而链化的条件有所不同,只有当链表长度小于6的时候,才会将红黑树重新转化为链表

示意图:

那为什么选取8、6这样的阈值呢?

应用官方的说明:

       * bin中的节点遵循泊松分布
       * (http://en.wikipedia.org/wiki/Poisson_distribution) 带有
       * 默认调整大小的参数平均约为 0.5
       * 阈值为 0.75,尽管由于以下原因存在较大方差
       * 调整粒度。 忽略方差,预期
       * 列表大小 k 的出现次数为 (exp(-0.5) * pow(0.5, k) /
       *阶乘(k))。 第一个值是:
       *
       * 0: 0.60653066
       * 1:0.30326533
       * 2:0.07581633
       * 3:0.01263606
       * 4:0.00157952
       * 5:0.00015795
       * 6:0.00001316
       * 7:0.00000094
       * 8:0.00000006

而链化选取6为链化阈值,是为了防止链表长度在7、8之间来回,从而导致不断链化、树化,浪费资源。

以上为put()方法的整个核心过程,下面上源码:

 public V put(K key, V value) {
                 // 对key的hashCode()做hash
         return putVal(hash(key), key, value, false, true);
 }
 ​
 final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) {
       Node<K,V>[] tab; Node<K,V> p; 
       int n, i;
       //table为空,先初始化
       if ((tab = table) == null || (n = tab.length) == 0)
           n = (tab = resize()).length;
       //计算hash对应的位置,null特殊处理
       if ((p = tab[i = (n - 1) & hash]) == null)
           tab[i] = newNode(hash, key, value, null);
       else {
           Node<K,V> e; K k;
           //如果节点已经存在就替换old value
           if (p.hash == hash &&
               ((k = p.key) == key || (key != null && key.equals(k))))
               e = p;
           //bucket中是红黑树
           else if (p instanceof TreeNode)
               e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
           //bucket中是链表
           else {
               for (int binCount = 0; ; ++binCount) {
                   if ((e = p.next) == null) {
                       p.next = newNode(hash, key, value, null);
                       //链表长度达到设定的转换为红黑树的阈值
                       if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                           //注意:这个方法并不是直接将链表转换为红黑树,如果当前buckets长度小于64,则扩容,否则将链表转换为红黑树
                           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;
       //当前容量超出 loadfactor*current capacity,则扩容
       if (++size > threshold)
           resize();
       afterNodeInsertion(evict);
       return null;
   }

三、HashMap的扩容

HashMap 中扩容方法为resize()。具体源码不作分析,主要阐述扩容的理念。

HashMap针对于数组长度达到负载因子*数组长度的时候,会对数组进行2倍扩容。扩容后,会对数据进行重新分布,这一步是消耗性能的地方。

扩容后数组元素的位置:

  • 无链

     if (e.next == null)
        newTab[e.hash & (newCap - 1)] = e;

    当数组下标有元素但未成链,则该元素按照newTab[e.hash & (newCap - 1)]的位置来存储。 扩容前:
    扩容后

  • 总结:HashMap 扩容后,原来的元素,要么在原位置,要么在原位置+原数组长度 那个位置上。

  • 有链 盗图:

    扩容后的元素位置根上面的总结一致。

四、Hash(key)方法

源码:

     static final int hash(Object key) {
         int h;
         return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
     }

先根据 key 的值计算到一个 hashCode,将 hashCode 的高 16 位二进制和低 16 位二进制进行异或运算,得到的结果再与当前数组长度减一进行与运算。最终得到一个数组下标,过程如下:

int hashCode = key.hashCode()

int hash = hash(key) = key.hashCode()的高 16 位^低 16 位 &(n-1) 其中 n 是当前数组长度

同时在这里要提醒一点

在 JDK1.7 和 JDK1.8 的时候对 hash(key)的计算是略有不同的

JDK1.8 时,计算 hash(key)进行了两次扰动

JDK1.7 时,计算 hash(key)进行了九次扰动,分别是四次位运算和五次异或运算

其中扰动可能理解为运算次数

示例:

 map.put("test","1");
 ​
 h=key.hashCode()    0000 0000 0011 0110 0100 0100 1001 0010
 h>>>16              0000 0000 0000 0000 0000 0000 0011 0110
                     ---------------------------------------
 hash = h^(h>>>16)   0000 0000 0011 0110 0100 0100 1010 0100
 (n-1)               0000 0000 0000 0000 0000 0000 0000 1111
 (n-1)& hash                                    0100 = 4

这里就可以解释一个HashMap中的现象---HashMap的容量一直都是2的倍数。

why?

其实这里的主要目的是为了减少Hash冲突(key计算的数组下标相同)。

当参与hash(key)算法的 (n-1) 的值尽可能都是1的时候,得到的值才是离散的。

参考文章: 深度解析 HashMap 底层实现架构 - 掘金 (juejin.cn)

[HashMap源码学习之路]---数组扩容后元素的前后变化_hashmap扩容后的位置-CSDN博客

java8-HashMap源码分析_java8 hashmap源码、-CSDN博客

标签:分析,hash,HashMap,链表,key,0000,null,HasMap,底层
From: https://blog.csdn.net/m0_74917967/article/details/136984671

相关文章

  • 【数据分析实战】物流行业数据分析
    数据来源:某企业销售的6种商品对应的送货及用户反馈数据解决问题:配送服务质量是否存在问题是否存在尚有潜力的销售区域商品是否存在质量问题分析过程:数据清洗①重复值、缺失值、格式调整②异常值处理(比如:消费金额存在等于0的情况,属于异常等)数据规整比如:增加一列辅......
  • 【数据分析实战】餐厅订单数据分析
    今天我们来分析以下某餐厅8月份订单数据,该餐厅的订单数据前10天、中间10天、后10天分别放在不同的Sheet里。订单数据字段包括:detail_id、order_id、dishes_id、logicprn_name、parent_class_name、dishes_name、itemis_add、counts、amounts、cost、piece_order_time、emp_......
  • js一些底层
    简介:JavaScript是一种高级编程语言,通常在网页开发中用于前端和后端开发。JavaScript的底层实现是浏览器或服务器上的JavaScript引擎。不同的引擎可能有不同的底层实现,但它们都有一个共同的目标,即执行JavaScript代码。JavaScript的底层实现涉及到多个方面,包括解释器、......
  • 量化交易入门(十六)Backtrader、Zipline、PyAlgoTrade对比分析
    量化交易发展这么多年,有很多优秀的前辈为我们提供了各种开源的交易回测系统,我将对常用的Backtrader、Zipline、PyAlgoTrade这三个量化交易回测平台进行详细介绍,并进行对比分析。一、Backtrader概述:Backtrader是一个Python的事件驱动型回测框架,由社区驱动开发,功能全面且灵......
  • [矩阵分析] 一、线性空间与线性变换
    目录线性空间及其性质定义关键性质1.封闭性2.加法运算的性质3.标量乘法的性质4.线性组合、跨度和线性独立性5.子空间、基和维数6.核和像核(Kernel)像(Image)核与像的关系应用一些重要的性质通俗理解线性空间的维数、基与坐标维数(Dimension)基(Basis)坐......
  • 关于使用PZ6808L开发板,调试USART3的问题分析
    首先,写代码方面相信,大家都可以搞定,网上也有很多人写的程序,这里关于如何驱动USART3,就不进行赘述了。关于这款开发板RS232模块,是给F4使用的,但是他留了两个接线柱,就是F1的USART3的两个接口。接下来就是接线的问题,如下图,将这个4个接线柱,两两交叉进行连接,跳线帽肯定搞不了,如下图......
  • gpio子系统分析
    参考博客:https://blog.csdn.net/yangguoyu8023/article/details/121892008https://blog.csdn.net/yangguoyu8023/category_11576708.html gpiolib相关数据结构:数据结构主要定义在include/linux/gpio/driver.h和/drivers/gpio/gpiolib.h中/***structgpio_chip-a......
  • Python 潮流周刊第 43 期(摘要),赠书 5 本《Python数据结构与算法分析(第3版)》
    本周刊由Python猫出品,精心筛选国内外的250+信息源,为你挑选最值得分享的文章、教程、开源项目、软件工具、播客和视频、热门话题等内容。愿景:帮助所有读者精进Python技术,并增长职业和副业的收入。周刊全文:https://pythoncat.top/posts/2024-03-23-weekly特别提醒:本期赠书5......
  • 【OpenFeign】@FeignClient 代理对象的创建源码分析
    1 前言我们从上节 【OpenFeign】@FeignClient注入过程源码分析 继续,来看看它代理对象的创建,以及请求的执行过程。我们就从它的 FeignClientFactoryBean看起,那我们这里简单回忆下它都设置了哪些属性,我简单画了个图。这些属性不了解的话,就先看看上节哈,有详细的说明,我这里......
  • 《数据结构与算法分析》作业一(详解版)
    1.什么是数据结构?有关数据结构的讨论涉及哪三个方面?答:(1)数据结构是指相互之间存在着一种或多种关系的数据元素的集合和该集合中数据元素之间的关系组成.(2)逻辑结构、存储结构以及运算结构。2.什么是算法?算法的特性有哪些?根据这些特性,解释算法与程序的区别?答:(1)算法是一组明......