Android基础知识-HashMap的Hash是什么?

首先,Hash 是 哈希算法,可以将一个复杂的值计算成一个数字,通常情况下,不同值的哈希值是不同的。如果出现不同值的哈希值相同,就是出现了哈希冲突。

本文关注的重点是,HashMap 的 hash 方法源码:

/**
     * Computes key.hashCode() and spreads (XORs) higher bits of hash
     * to lower.
     * 计算 key.hashCode() 并且将高位与地位进行异或运算。
     * Because the table uses power-of-two masking, sets of
     * hashes that vary only in bits above the current mask will
     * always collide.
     * 由于 map 的表格使用使用 2 的 n 次幂作为掩码,仅在掩码位之上发生变化的哈希集合频繁出现哈希碰撞。
     * (因为 map 使用数组存储数据,数组的长度总是 2 的 n 次幂,通常情况下数组的长度比较小,只在比较高位变化的 hash 与
     * 数组长度取模得到的结果总是相同的,所以需要将高位与低位异或运算,这样可以将高位的影响传递给低位,减少 hash 冲突)
     * (Among known examples are sets of Float keys
     * holding consecutive whole numbers in small tables.)
     * (已知的示例中有一组 Float 键值,它们在小表中持有连续的整数。)
     * 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
     * more: less than 1 in ten million
     * 更多的小于1的8位小数
     * So we apply a transform that spreads the impact of higher bits
     * downward.
     * 因此,我们应用了将高位向下影响的变换。
     * There is a tradeoff between speed, utility, and quality of bit-spreading.
     * 在速度,实用性和位扩展质量之间进行权衡
     * Because many common sets of hashes are already reasonably distributed
     * (so don't benefit from spreading), and because we use trees to handle large sets of
     * collisions in bins, we just XOR some shifted bits in the
     * cheapest possible way to reduce systematic lossage, as well as
     * to incorporate impact of the highest bits that would otherwise
     * never be used in index calculations because of table bounds.
     * 因为很多常见的哈希集已经合理的分布(所以不能从传播中受益),而且我们使用了树结构处理箱子中大量的冲突
     * 我们仅使用移位后的位运算这种最廉价的方式减少系统损失,以合并最高位的影响,
     * 否则由于表范围的限制,这些位将不永远不会在索引计算中使用。
     */
    static final int hash(Object key) {
        int h;
        return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
    }

可以看到方法非常简单,只是使用 key.hashCode() 获取哈希值,然后与自己的高 16 位进行异或运算并返回。

此处插播一个没啥用的知识点,当 key 是 0 的时候,哈希值为 0,也就是说 HashMap 可以存储 key 为 null 的值,并且放在数组首位,与其他值的 key 使用起来区别不大。恭喜你,学到了一个没有用的知识点,可以面试的时候给面试官显摆一下。别问我为啥关注这个,因为面试官问过我这个[弱鸡]。

然后,可以看下 key.hashCode() 的源码。

public native int hashCode();

没错只有一行,它是个原生方法,具体实现我也不了解,这不是本文的重点。因为 hash 方法需要被频繁使用,put、get 等方法都会使用,所以它必须执行起来非常迅速,所以,这是使用原生方法的一个原因。其次,hash 方法的另一个需求就是尽量减少哈希冲突的情况,那个与自己高 16 位异或计算就是低成本的避免哈希冲突的一个思路,在我的塑料英语翻译中能看见详细的描述。

by 费城的二鹏 2020.07.16 长春

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