HashMap源码-概述

Implementation Notes

  • HashMap有两个参数,initialCapacity(默认16),loadFactor默认0.75,当容器内节点数量多于initialCapacity*loadFactor,自动扩充
  • loadFactor越大,时间(puth和get的时间)成本越高,越小,空间成本越高
  • 一般情况下,内部存储的是哈希表,当内容过大时,转变为TreeNode容器,TreeNode容器内部类似TreeMap的结构,HashMap的方法大部分不区分,只在TreeNode有额外实现时被调用(通过instrance of TreeNode)方法,使用TreeNode容器是为了在数据过多时能够快速查找,然而因为大部分Map内部元素没有达到需要Tree Bin(树状容器)存储的要求(默认64个元素),所以checking for existence of tree bins may be delayed in the course of table methods.
  • 树形态根据key的哈希码排序,但是当类实现了Comparable接口时,类的compareTo方法被调用来排序,在多个实例返回同样的hashCode的情况下,通过实现compareTo方法来提高效率
  • TreeNode大小大概是普通Node的两倍,因此需要一个阈值来控制它的启用

常量

  • DEFAULT_INITIAL_CAPACITY 初始化容量,16
  • MAXIMUM_CAPACITY 最大容量,2的30次方
  • DEFAULT_LOAD_FACTOR 负载比例,0.75
  • TREEIFY_THRESHOLD 树状阈值,HashMap相同hash的默认放到同一个节点并next链时穿起来,但当链长>=TREEIFY_THRESHOLD时,需将链变为树状以提升访问效率
  • UNTREEIFY_THRESHOLD (树状转数组的阈值,6)
  • MIN_TREEIFY_CAPACITY 最小转树状阈值,64

内部静态Node类

  • 内部存储final hash,final key,value,next四个变量
  • hashCode方法返回Objects.hashCode(key) ^ Objects.hashCode(value)
public final int hashCode() {
            return Objects.hashCode(key) ^ Objects.hashCode(value);
        }
  • equals方法要求key和value对equals方法都成立

方法

1.hash方法

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

计算key的hash,结尾是key的hashCode方法返回值(int型)的高16位拼接高16位与低16位按位异或的结果
这么做的原因是:table中计算key对应存储位置的时候,使用的是(capacity-1)&hash,当n=16时,相当于16&hash,也就是,000000000000000001111&hash,也就是前28位确认为0,hash的后四位决定是否碰撞,对于只在高位有区别的key,是大概率会碰撞的,因此将高16位spread到低位去,可以在某些场景减少碰撞,下面是验证:

public static void main(String[] args) {
        Float f1 = 11111.0f;
        Float f2 = 111111.0f;
        int capacity = 16;
        System.out.println("f1的二进制:"+Integer.toBinaryString(f1.hashCode()));
        System.out.println("f2的二进制:"+Integer.toBinaryString(f2.hashCode()));
        System.out.println("HashMap630行,位置计算方式为(n - 1) & hash,假设当前容量为16\n若不进行spread,则:");
        int index1 = (capacity-1)&f1.hashCode();
        int index2 = (capacity-1)&f2.hashCode();
        System.out.println("位置1为:"+index1);
        System.out.println("位置2为:"+index2);
        System.out.println("发生碰撞\n若进行spread:");
        index1 = (capacity-1)&spreadHash(f1.hashCode());
        index2 = (capacity-1)&spreadHash(f2.hashCode());
        System.out.println("位置1为:"+index1);
        System.out.println("位置2为:"+index2);
        System.out.println("不发生碰撞");
    }

    /**
     * 按hashMap的方法,计算spread后的hash
     * @param hash
     * @return
     */
    static  int spreadHash(int hash){
        return hash ^ (hash >>> 16);
    }

f1的二进制:1000110001011011001110000000000
f2的二进制:1000111110110010000001110000000
根据HashMap源码630行,位置计算方式为(n - 1) & hash,假设当前容量为16
若不进行spread,则:
位置1为:0
位置2为:0
发生碰撞
若进行spread:
位置1为:13
位置2为:9
不发生碰撞

为什么是^不是|或者&,验证可得按&是不好的,比如上面例子,按位与的话,还是会碰撞(结果都是0),^和|的运算结果一致,待探究

2.comparableClassFor方法:判断是否实现Comparable接口

static Class<?> comparableClassFor(Object x)

  • 因为String类型的key最多,且实现了Comparable接口,所以入参为String类的直接返回String.class,
  • 判断类实现Comparable接口,且泛型实际类型为自己,则返回
  • 否则返回null

3.compareComparables方法:返回两个Comparable对象的比较值

static int compareComparables(Class<?> kc, Object k, Object x)

  • kc:实现了Comparable接口的实体类
  • k:对象1 c:对象2
  • 通过调用时保证类型正确
  • x为空或者不是kc类型时返回空,否则返回k.compareTo(x)

4.tableSizeFor:HashMap的方法,计算目标容量对应的2的次方数容量

static final int tableSizeFor(int cap)

内部变量

1.transient Node<K,V>[] table;

核心存储变量,随需要resize,初始化时,容量大小是2的次方

2.transient Set<Map.Entry<K,V>> entrySet;

保存缓存的entrySet

3.transient int size;

存储当前map的大小

4.transient int modCount;

存储当前map发生Structural modifications的次数,用于在迭代时快速失败(fail-fast)

5.int threshold;

下一次resize的阈值,比如当前threshold为16,当前要放17个元素进去,则需要resize

6.final float loadFactor;

负载因子

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

推荐阅读更多精彩内容

  • 亲爱的家长您好! 又一周过去了! 本周依然是语文版块,我们继续享受故事和汉字的工作,故事时间总是孩...
    0懒亮亮0阅读 283评论 0 1
  • 听说,孤独是一个人的狂欢。其实,欢乐永远和孤独者无关。 夜色渐重,黑暗趁人不备将我重重地包围了起来。 浓重的黑暗,...
    听雨Q晴阅读 114评论 0 2
  • 从你的专业角度来说,你所见过的最糟糕的建议是什么?“追随你的梦想。”没有多年的自我认知,你根本不可能做到这点。你是...
    紫苑阅读 1,027评论 5 18