百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

HashMap详解 hashmap理解

yuyutoo 2024-10-12 01:23 2 浏览 0 评论

讲解步骤

  • 基础知识
  • 工作原理
  • 关键代码
  • 核心方法

基础知识

数组结构

数组接口,在查询数据方面,具备优势

链表结构

链表结构,在增删数据方面,具备优势

红黑树结构

红黑树结构,在查询数据方面,数据量较大的时候,具备一定的优势

什么是散列(哈希)表

散列表,顾名思义,就是将数据分布在不同的列
但是散列表并不是完全将数据分散在不同的列,而是按照某种规则,将具备同样规则的数据存储在同一列。
即具备相同规则的数据存储在同一列,规则不同的数据分布在不同的列。
这种规则最终的产生与哈希值有关。
这里需要注意的事,哈希值只是确定最后存储列的因素,也就是说不同的哈希值可能会存在同一列。

什么是哈希值

哈希值简单的说,就是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;
    }



相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: