HashMap之流程分析

HashMap源码分析

HashMap其中最关键的就是他的put方法,你如果真的理解了put方法的全过程,hashMap你可能就理解了大半。

1. Put流程认识全过程

先将put的源码贴出来了,方面大家进行对照。

在这里HashMap的put流程我先不把绘制的流程图贴出来,自己进行debug的流程来分析。 HashMap,顾名思义,肯定是和hash表有关系的。接来下先看看 HashMap的hash函数。

HasMap debug流程

调用普通方法,传入key 和 value,调用 一个 hash方法

hash方法

这段代码的意思就是:当key==null返回0,负责调用hasCode方法和他本身的低16进行一个异或操作。拿到我们的第一个 hash值可能不太理解为什么这么做,请看本专栏的Hash函数部分会有相应解答。这里我们就将他当成一个特殊的哈希函数就可以。(因为上面已经贴出图片,下面统一使用代码块的方式进行)

putval方法的第一步

java 复制代码
Node<K,V>[] tab; Node<K,V> p; int n, i;
if ((tab = table) == null || (n = tab.length) == 0)
    n = (tab = resize()).length;

Node数组就是HashMap存储数据的容器。第一个if,如果 Node数组为null或者,Node数组长度为0,对数组进行扩容。

第二步

java 复制代码
if ((p = tab[i = (n - 1) & hash]) == null)
    tab[i] = newNode(hash, key, value, null);

(n-1)&hash 他的含义和 hash%n是一样的 ,n是数组长度,这里我认为,这是HashMap第二个hash函数,到这里我们需要理解:通过第一个高低16位异或操作,拿到我们第一个Hash值,在从一个hash值通过第二个hash函数定义为 Node数组中的位置。当前位置为null,直接插入元素。

第三步

java 复制代码
else { ,,,,}

前两个条件都不符合,看看他是如何操作的。

  1. 总是检查第一个
java 复制代码
if (p.hash == hash &&
    ((k = p.key) == key || (key != null && key.equals(k))))
    e = p;

在HashMap的源码中,它总是检查第一个元素, 主要代码: (k=p.key)==key,当前节点的key和传入的key如果相等,先记录当前节点。

java 复制代码
else if (p instanceof TreeNode)
    e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);

如果说他不在第一个元素上,判断当前节点是不是树,如果是树把当前节点插入树中。(具体细节不详细解读,可以自行观看)。

java 复制代码
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 //插入玩之后判断是否到达树化的条件,到达树化的条件,进行树化。
                treeifyBin(tab, hash);
            break;
        }
        //判断链表中有没有 相同的key值,如果有直接跳出循环。
        if (e.hash == hash &&
            ((k = e.key) == key || (key != null && key.equals(k))))
            break;
        p = e;  // 使链表前移。
    }
}

最后,上面的情况都不符合,就到了链表的情况, 总结大概流程:遍历链表

  • 遍历链表中:如果有相同的key,不做任何操作
  • 如果链表中没有相同的key,将该元素插入链表尾部,之后判断是否达到树化的条件,如果达到树化的条件,进行树化操作。
java 复制代码
if (e != null) { // existing mapping for key
    V oldValue = e.value;
    if (!onlyIfAbsent || oldValue == null)
        e.value = value;
    afterNodeAccess(e);
    return oldValue;
}

仔细分析,e!=null的情况是,链表中有相同的key的情况和第一个元素key相同时,总之,就是有相同Key的时候,我们去覆盖 对应的value值,不做任何操作。

java 复制代码
++modCount;
if (++size > threshold)  //
    resize();
afterNodeInsertion(evict);
return null;

最后,我们要判断是否达到扩容条件,如果达到扩容条件,就要进行扩容。接下来就自然而然能画出标准的流程图。

###总结:流程图

相关推荐
用户83562907805128 分钟前
Python 实现 PDF 文件加密与解密方法
后端·python
小满zs31 分钟前
Go语言第二章(小无相功)
后端·go
用户83562907805132 分钟前
使用 Python 冻结与拆分 Excel 窗格教程
后端·python
karry_k37 分钟前
MyBatis批量insert-select踩坑:useGeneratedKeys=true 可能让PostgreSQL返回大量插入结果
java·后端
妙码生花40 分钟前
从 PHP 到 AI + Golang,程序员自救转型手记(十九):点选验证码代码逐行目检
前端·后端·go
贰先生44 分钟前
Xiuno BBS X版 用户封禁系统
后端
karry_k44 分钟前
PostgreSQL 在 MyBatis 中执行正常 SQL 失效:一次 DELETE USING 踩坑记录
java·后端
ServBay1 小时前
不会写代码也能建站?AI 时代,非技术创始人如何从零搭建自己的 Web 项目
后端·mcp
Moladev1 小时前
如何在 Electron 中接入 OpenAI 兼容的大模型 API:Snaptium 的主进程代理实践
后端
Oneslide1 小时前
根分区爆满却找不到大文件?深度解析 Linux df 与 du 不一致的经典故障
后端