HsahMap、ConcurrentHashMap

个人学习总结整理
资料来源:https://www.itcodemonkey.com/article/1266.html
via:程序员小灰(VX公众号)

HashMap

- 实现原理及冲突解决

HashMap是一个用于存储Key-Value键值对的集合,每一个键值对也叫做Entry。这些个键值对(Entry)分散存储在一个数组当中,数组中每一个元素的初始值都是Null。
执行Put操作时,通过对key键进行hashcode的计算确定其在数组中的位置,当哈希值发生冲突时,则通过链表来解决冲突。HashMap数组的每一个元素不止是一个Entry对象,也是一个链表的头节点。每一个Entry对象通过Next指针指向它的下一个Entry节点。当新来的Entry映射到冲突的数组位置时,只需要插入到对应的链表即可。
index = HashCode(Key) & (Length - 1)
HashMap的默认长度为16,且必须为2的幂。

- 高并发下的HashMap

Capacity:HashMap的当前长度。上一期曾经说过,HashMap的长度是2的幂。
LoadFactor:HashMap负载因子,默认值为0.75f。
当 HashMap.Size >= Capacity * LoadFactor 时,HashMap容量饱和,将发生ReSize:

 1. 扩容(创建一个长度为原来数组二倍的新数组)
 2. ReHash(遍历原Entry数组,把所有的Entry重新Hash到新数组)

ReHash代码:

/**
 * Transfers all entries from current table to newTable.
 */
void transfer(Entry[] newTable, boolean rehash) {
    int newCapacity = newTable.length;
    for (Entry<K,V> e : table) {
        while(null != e) {
            Entry<K,V> next = e.next;
            if (rehash) {
                e.hash = null == e.key ? 0 : hash(e.key);
            }
            int i = indexFor(e.hash, newCapacity);
            e.next = newTable[i];
            newTable[i] = e;
            e = next;
        }
    }
}

高并发下容易出现环形链表

ConcurrentHashMap

- 实现原理

Segment本身就相当于一个HashMap对象。有2的N次方个,共同保存在一个名为segments的数组当中。
这里写图片描述

Get方法:
1、为输入的Key做Hash运算,得到hash值。
2、通过hash值,定位到对应的Segment对象
3、再次通过hash值,定位到Segment当中数组的具体位置。

Put方法:
1.为输入的Key做Hash运算,得到hash值。
2、通过hash值,定位到对应的Segment对象
3、获取可重入锁
4、再次通过hash值,定位到Segment当中数组的具体位置。
5、插入或覆盖HashEntry对象。
6、释放锁。

Size方法:
1、遍历所有的Segment。
2、把Segment的元素数量累加起来。
3、把Segment的修改次数累加起来。
4、判断所有Segment的总修改次数是否大于上一次的总修改次数。如果大于,说明统计过程中有修改,重新统计,尝试次数+1;如果不是。说明没有修改,统计结束。
5、如果尝试次数超过阈值,则对每一个Segment加锁,再重新统计。
6、再次判断所有Segment的总修改次数是否大于上一次的总修改次数。由于已经加锁,次数一定和上次相等。
7、释放锁,统计结束。
为了尽量不锁住所有Segment,首先乐观地假设Size过程中不会有修改。当尝试一定次数,才无奈转为悲观锁,锁住所有Segment保证强一致性。

- ConcurrentHashMap JDK1.7和1.8有什么区别
https://www.jianshu.com/p/933289f27270
https://blog.csdn.net/programmer_at/article/details/79715177

黑红树

HashMap中,在链表长度大于8的时候,将后面的数据存在红黑树中,通过平衡来二分查找,允许局部很少的不完全平衡,以加快检索速度。

特征:
1)每个节点或者是黑色,或者是红色。
2)根节点是黑色。
3)每个叶子节点(NIL/NULL)是黑色。
4)如果一个节点是红色的,则它的子节点必须是黑色的。
5)从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值