HashMap源码解析

Java 8系列之重新认识HashMap
关于HashMap,上面链接里美团团队出的文章已经很好了。这篇博客详细聊一下HashMap里几个关键的算法。

二次hash算法

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

源码只有一行,分为3步:

  • h = key.hashCode(),取到哈希码
  • h >>> 16,哈希码无符号右移16位(原长度32位),高位补0。这一步相当于把哈希码的高16位变成低位。
  • h 和右移的结果做异或。由于h >>> 16后高16位为0,任何数和0异或都是其本身,所以这一步保证了二次hash的结果中,高16位不变,低16位有key.hashCode()高位和低位的特征。

为什么要这样做呢?和hashMap中下标计算有关,往下看。

取下标算法

n = table.length;
index =  (n-1) & hash;

由于n = table.length必为2的x次幂,(n-1) & hash时:

  • 只要x <= 16即数组长度小于65535时,hash中只有低16位参与了运算。所以二次hash算法中低位结合了高位的特征,就是为了减小hash碰撞。
  • (n-1) & hash等效于hash / n,即求余操作。举个栗子,n=16时,如下:
hash值   10101010
n-1      00001111
——————————————————
&        00001010  // 和1与时不变,和0与时=0

通过位运算实现了等效于求余的算法,效率更高。

tableSizeFor()

把构造函数传入的表示长度的任意数字优化为最接近它的2的n次幂的值,比如传入6,最终为8,传入129,最终为256。

    static final int MAXIMUM_CAPACITY = 1 << 30;
    /**
     * Returns a power of two size for the given target capacity.
     */
    static final int tableSizeFor(int cap) {
        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;
    }

下面分析这个算法
首先,为什么要对cap做减1操作。int n = cap - 1;
这是为了防止,cap已经是2的幂。如果cap已经是2的幂, 又没有执行这个减1操作,则执行完后面的几条无符号右移操作之后,返回的capacity将是这个cap的2倍。如果不懂,要看完后面的几个无符号右移之后再回来看看。
下面看看这几个无符号右移操作:
如果n这时为0了(经过了cap-1之后),则经过后面的几次无符号右移依然是0,最后返回的capacity是1(最后有个n+1的操作)。
这里只讨论n不等于0的情况。
第一次右移

n |= n >>> 1;

由于n不等于0,则n的二进制表示中总会有一bit为1,这时考虑最高位的1。通过无符号右移1位,则将最高位的1右移了1位,再做或操作,使得n的二进制表示中与最高位的1紧邻的右边一位也为1,如000011xxxxxx。
第二次右移

n |= n >>> 2;

注意,这个n已经经过了n |= n >>> 1; 操作。假设此时n为000011xxxxxx ,则n无符号右移两位,会将最高位两个连续的1右移两位,然后再与原来的n做或操作,这样n的二进制表示的高位中会有4个连续的1。如00001111xxxxxx 。
第三次右移

n |= n >>> 4;

这次把已经有的高位中的连续的4个1,右移4位,再做或操作,这样n的二进制表示的高位中会有8个连续的1。如00001111 1111xxxxxx 。
以此类推
注意,容量最大也就是32bit的正数,因此最后n |= n >>> 16; ,最多也就32个1,但是这时已经大于了MAXIMUM_CAPACITY ,所以取值到MAXIMUM_CAPACITY
举个栗子,如果构造函数里传入的数组长度是13,二进制为1101,经过n |= n >>> 1后还是1101,再n |= n >>> 2,为11111111后续右移4、8、16都为0000,再求或还是它本身。最后return n+1,即为1111 + 1 = 10000,即十进制的16,大功告成。

resize() 扩容算法

java 8 的扩容算法因为引入了红黑树,所以比较复杂,我们只提其中比较关键的地方:扩容后怎么确认新下标?
首先要知道扩容是2倍扩容,长度依然为2的次幂。
常规思路当然是重走一次index = (table.length-1) & hash,java 8之前确实是这样做的。

java8优化了这个算法:只需要看看原来的hash值新增的那个bit是1还是0就好了,是0的话索引没变,是1的话索引变成“原索引+oldCap,即hash & oldCap == 0”。
上面这句话是啥意思?举个例子:

// 假设hash值为10101010,原数组长度为8,扩容后为16,根据(table.length-1) & hash 即为
         10101010
oldCap-1 00000111
——————————————————
  &      00000010

         10101010
newCap-1 00001111
——————————————————
  &      00001010
//  oldIndex即为hash前三位的值010,newIndex是hash前四位的值1010。
//  关键在新增的这一位,为0时 oldIndex=newIndex,位置不变;为1时newIndex = oldIndex + oldCap
// 所以这里无需再计算(table.length-1) & hash,只要判断新增的这一位是0还是1就可以;

         10101010
oldCap   00001000
——————————————————
  &      00001000
// 直接hash & oldCap == 0,true 时新增的bit为0;index不变,false 时为1,newIndex = oldIndex + oldCap

总结

所以HashMap的长度为什么要优化为2的次幂?从上面的算法可以看出:

  1. 取下标算法(table.length-1) & hash ,当length为2的次幂时,等效于求余,比求余效率更高。
  2. 扩容时只需要判断新增的那个bit即可确定下标。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,723评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,485评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,998评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,323评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,355评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,079评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,389评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,019评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,519评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,971评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,100评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,738评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,293评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,289评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,517评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,547评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,834评论 2 345

推荐阅读更多精彩内容

  • 原文地址 HashMap HashMap 是 Map 的一个实现类,它代表的是一种键值对的数据存储形式。 大多数情...
    gyl_coder阅读 373评论 0 0
  • 适合阅读的人群 文笔不是很好,如果对于HashMap没有一个大致的了解,最好不要看 目录 HashMap的几个基本...
    K807阅读 245评论 0 2
  • ref1: http://blog.csdn.net/fan2012huan/article/details/51...
    不存在的里皮阅读 154评论 0 0
  • 前言 HashMap使我们经常使用的数据结构,大家常说map的数据结构就是数组+链表,其中数组指的就是HashMa...
    冠冠从小爱学习阅读 240评论 0 1
  • 昨天下午在七巧板张老师的帮助下做了有生以来第一次沙盘,路上有紧张,有兴奋,想着我会摆个什么样的沙盘。到了地方开始摆...
    欣宝_79e5阅读 291评论 0 0