Java1.8 hashmap 源码阅读1

建议先看下文章中提供的源码,然后再看解释可以加深理解。

内部静态变量

DEFAULT_INITIAL_CAPACITY

默认初始化容量

DEFAULT_LOAD_FACTOR

默认负载因子

TREEIFY_THRESHOLD

二叉树阈值

UNTREEIFY_THRESHOLD

取消二叉树阈值

MIN_TREEIFY_CAPACITY

二叉树化所需要的最小容量

内部类,Node<K, V>

好像内部存储的hash没用到。

hashCode

计算hash值,

public final int hashCode() {
    return Objects.hashCode(key) ^ Objects.hashCode(value);
}


静态方法

static final inti hash(Object key)

计算key的hash值。

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

static Class<?> comparableClassFor(Object x)

TODO

在二叉树中使用到

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

TODO

也是在二叉树中用到

static final int tableSizeFor(int cap)

传入容量cap,返回比传入的容量cap大的最小二次幂。

变量

table

最终存储数据的地方

entrySet

TODO

一个缓存用的set

size

容量大小当前

modCount

TODO

操作数,貌似是线程安全用的

threshold

resize的阈值

loadFactor

负载因子

公有方法

构造方法

public HashMap(int initialCapacity, float loadFactor)
public HashMap(int initialCapacity)
public HashMap()
public HashMap(Map<? extends K, ? extends V> m)

pubMapEntries(Map<? extends K, ? extends V> m, boolean evict)

一次性放入多组键值对,也可以叫做键值对的复制。

  • 如果table没有初始化则进行初始化,根据传入的m计算最大容量(根据负载因子)
  • 如果table初始化了,则判断是否会大于阈值,大于阈值则resize
  • for遍历m,完成数据的复制

注意:putVal函数

public int size()

返回map的容量

public boolean isEmpty()

返回map是否为空

public V get(Object key)

通过调用内部方法getNode来获取数据

final Node<K, V> getNode(int hash, Object key)

final Node<K,V> getNode(int hash, Object key) {
    Node<K,V>[] tab; Node<K,V> first, e; int n; K k;
    if ((tab = table) != null && (n = tab.length) > 0 &&
        (first = tab[(n - 1) & hash]) != null) {
        if (first.hash == hash && // always check first node
            ((k = first.key) == key || (key != null && key.equals(k))))
            return first;
        if ((e = first.next) != null) {
            if (first instanceof TreeNode)
                return ((TreeNode<K,V>)first).getTreeNode(hash, key);
            do {
                if (e.hash == hash &&
                    ((k = e.key) == key || (key != null && key.equals(k))))
                    return e;
            } while ((e = e.next) != null);
        }
    }
    return null;
}
  • 如果table为null或者长度为0,或者计算的index的对应值为null,则返回null(没有匹配)
  • 否则检查第一个值,如果keys也相等则说明找到,返回
  • 否则检查是不是TreeNode,如果是则调用getTreeNode
  • 否则遍历链表后找到值后返回

public boolean containsKey(Object key)

调用内部方法getNode,判断是否包含key

public V put(K key, V value)

调用内部方法putVal来实现

final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict)

final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
               boolean evict) {
    Node<K,V>[] tab; Node<K,V> p; int n, i;
    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) {
                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;
}

很重要,用来往table里放东西的方法。

  • 检查table,必要时做初始化(resize)
  • 计算index,如果不和任何hash冲突,则新建Node
  • 否则查看是否是同一个key(hash也相同),如果是则覆写
  • 否则发生撞库,判断是否是TreeNode,如果是则调用putTreeval方法
  • 否则遍历链表,如果在链表中有相同的hash和key,则覆写
  • 否则插入链表链尾,如果插入后超过阈值,则Treeify
  • 如果onlyIfAbsent是true的话,则不更新旧值
  • 如果size大于阈值,则调用resize

onlyIfAbsent: 只有当不存在时才更新

evict:

还有一些钩子函数

  • afterNodeAccess
  • afterNodeInsertion

final Node<K, V>[] 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;
}

resize是很重要的一个内部使用的函数,

主要是用来处理数据结构的初始化,以及在增删Node的时候,

通过resize来保持数据结构的合理性。

table为内部保存数据的数据结构。

流程的大概总结:

  • 计算当前容量,如果已经大于等于最大容量,则没必要resize,直接返回
  • 如果当前容量的两倍小于最大容量,并且大于初始容量,则当前阈值和容量翻倍
  • 如果容量为零,但是阈值不为零,则将容量设置为阈值的值
  • 否则的话则初始化容量和阈值
  • 如果阈值还是零的话,则用当前阈值和负载因子计算新的阈值

到此为止,阈值和容量的事情我们搞定了

下面开始把旧的table中的Node放到新的table中去。

  • 对于每一个table中的Node,计算新的index
  • 还是老样子,如果是单个节点,则直接放进新table中
  • 如果是TreeNode,则调用tree的方法,split
  • 否则插入链表,因为resize之后,每个元素都可能出现在不同的位置上,所以判断插入的位置,然后建立两个链表

判断方式:(e.hash & oldCap) == 0

解释:因为resize之后,左移一位,相当于最高位左移一位,因此如果e.hash & oldCap为零的话,

说明e.hash的高位上没有1,因此就算newCap和e.hash并运算结果也不会改变。

用此方法可以判断每个e到底属于哪个链表,

只可能有两种,一种是保持原来的链表,一种是进入新的链表。

final void treeifyBin(Node<K, V>[] tab, int hash)

final void treeifyBin(Node<K,V>[] tab, int hash) {
    int n, index; Node<K,V> e;
    if (tab == null || (n = tab.length) < MIN_TREEIFY_CAPACITY)
        resize();
    else if ((e = tab[index = (n - 1) & hash]) != null) {
        TreeNode<K,V> hd = null, tl = null;
        do {
            TreeNode<K,V> p = replacementTreeNode(e, null);
            if (tl == null)
                hd = p;
            else {
                p.prev = tl;
                tl.next = p;
            }
            tl = p;
        } while ((e = e.next) != null);
        if ((tab[index] = hd) != null)
            hd.treeify(tab);
    }
}

链表到树结构的转换

  • 如果传入的tab不符合要求,则resize
  • 否则,计算index找到对应元素,开始treeify
  • 调用内部方法replacementTreeNode新建一个TreeNode
  • 创建树头hd,之后对于新的节点,他的prev指向上一轮的旧节点,旧节点的next指向新节点
  • 创建红黑树,treeify,之后细讲

public void putAll(Map<? extends K, ? extends V> m)

调用内部的函数putMapEntries

public V remove(Object key)

继续封装,调用内部的removeNode方法。

返回被删除的元素

final Node<K, V> removeNode(int hash, Object key, Object value, boolean matchValue, boolean moveable)

final Node<K,V> removeNode(int hash, Object key, Object value,
                           boolean matchValue, boolean movable) {
    Node<K,V>[] tab; Node<K,V> p; int n, index;
    if ((tab = table) != null && (n = tab.length) > 0 &&
        (p = tab[index = (n - 1) & hash]) != null) {
        Node<K,V> node = null, e; K k; V v;
        if (p.hash == hash &&
            ((k = p.key) == key || (key != null && key.equals(k))))
            node = p;
        else if ((e = p.next) != null) {
            if (p instanceof TreeNode)
                node = ((TreeNode<K,V>)p).getTreeNode(hash, key);
            else {
                do {
                    if (e.hash == hash &&
                        ((k = e.key) == key ||
                         (key != null && key.equals(k)))) {
                        node = e;
                        break;
                    }
                    p = e;
                } while ((e = e.next) != null);
            }
        }
        if (node != null && (!matchValue || (v = node.value) == value ||
                             (value != null && value.equals(v)))) {
            if (node instanceof TreeNode)
                ((TreeNode<K,V>)node).removeTreeNode(this, tab, movable);
            else if (node == p)
                tab[index] = node.next;
            else
                p.next = node.next;
            ++modCount;
            --size;
            afterNodeRemoval(node);
            return node;
        }
    }
    return null;
}

参数好多。。

先说下参数

  • hash,对应的哈希值
  • key,对应的key值,这个是唯一的
  • value,对应的value
  • matchValue,配合value使用的,只删除value也相同的Node
  • moveable,用在tree中

具体的逻辑

  • 如果符合条件(table不为空,hash对应的位置有数据之类),则正式开始
  • 找到要删除的Node,和putVal差不多,单值,红黑树或者链表,找到为止
  • 如果找到的node不为空,且value相同(matchValue),则根据node类型删除
  • 如果是TreeNode,调用removeTreeNode方法
  • 如果是firstNode,则tabl[index] = node.next
  • 否则p.next = node.next,经典的链表删除元素的方法

钩子函数:afterNodeRemoval

public void clear()

清空所有的元素,tab[i] = null

public boolean containsValue(Object value)

因为每一个Node都有next属性,

所以对table遍历,再对node遍历即可找到是否包含value的node。

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

推荐阅读更多精彩内容

  • 1.HashMap是一个数组+链表/红黑树的结构,数组的下标在HashMap中称为Bucket值,每个数组项对应的...
    谁在烽烟彼岸阅读 1,004评论 2 2
  • 摘要 HashMap是Java程序员使用频率最高的用于映射(键值对)处理的数据类型。随着JDK(Java Deve...
    周二倩你一生阅读 1,238评论 0 5
  • 前言 这次我和大家一起学习HashMap,HashMap我们在工作中经常会使用,而且面试中也很频繁会问到,因为它里...
    liangzzz阅读 7,955评论 7 102
  • 一 两年前的今天 我从沙河跑到学院路 就站在某公寓楼下 看白杨高耸 柳树低垂 看冬青翠绿 银杏金黄 一抬头 那个身...
    自唯至熟阅读 554评论 0 0
  • 在做项目的时候,如果项目是前后分离的,后端一定要和前端或者是移动端对接接口,那么问题来了,接口是不是要自己写给他们...
    大猪大猪阅读 296评论 0 1