2/7ConcrruentHashMap和锁的总结

首先你要知道1.8的HashMap出现红黑树。所以在1.8的ConcurrentHashMap也出现了红黑树
1.几种扩容:
ArrayList初始10,1.5倍扩增。grow数组复制法。
HashMap初始16,2倍扩增,0.75的利用率。index=hash&(length-1)rehashing
HashTable(不常用)初始11,2倍+1扩增,0.75利用率。(奇数)index=hash%length。。rehashing
ConcurrentHashMap的扩容与HashMap一样
2.HashTable说说:
数组+链表的结构
加锁,synchronized
扩容的说一遍
3.HashMap与HashTable区别:(数组链表红黑树这个大前提不变)
默认容量不同,扩容不同。
因为加锁造成的效率不同。
线程安不安全。
4.ConcrruentHashMap(jdk1.7还没红黑树)
1.数据结构:Segment数组+HashEntry数组+链表
2.加锁方式:Reentrantlock(可重入and互斥锁)
value,以及链表都是 volatile 修饰的,保证了获取时的可见性。
3.put()方法
1.通过竞争获取锁,每个segment有自己的锁
2.通过key定位到segment
3.通过key的hashCode定位到HashEntry数组,值为null则添加,并判断是否需要扩容。
4.值若不为null,找到的是链表头节点,遍历找到相应的key,覆盖旧值。
5.解除锁
注:
首先第一步的时候会尝试获取锁,如果获取失败肯定就有其他线程存在竞争,则利用 自旋获取锁。
如果重试的次数达到了 MAX_SCAN_RETRIES 则改为阻塞锁,保证能获取成功。最后解除当前 Segment 的锁
5.CocurrentHashMap(JDK 1.8)
CAS + synchronized(保证原子性) 来保证并发安全性。其中的 val next 都用了 volatile 修饰,保证了可见性。
1.get方法:(不需要锁
根据计算出来的 hashcode 寻址,如果就在桶上那么直接返回值
如果是红黑树那就按照树的方式获取值
就不满足那就按照链表的方式遍历获取值
2.引入了CAS:
借助Unsafe类,CAS有3个操作数,内存值 V、旧的预期值 A、要修改的新值 B。当且仅当预期值 A 和内存值 V 相同时,将内存值V修改为 B,否则什么都不做。
3.CAS 会出现的问题:ABA
解决:对变量增加一个版本号,每次修改,版本号加 1,比较的时候比较版本号
4.put 过程0.75加载因子(Synchronized锁)

1.根据 key 计算出 hashcode,定位在table中的位置。
2.判断是否需要对table初始化,如果没有则初始化数组。(懒加载方式,第一次插入数据才初始化)
3.通过 key 定位出的 Node,如果为空表示当前位置可以写入数据,利用 CAS 尝试写入 ,失败则自旋保证成功
4.如果当前位置不为空,判断节点类型,如果该节点fh==MOVED(代表forwardingNode,数组正在进行扩容)的话,说明正在进行扩容。
5.如果都不满足,则说明我们找到的是这个链表头节点,依次向后遍历,有key就覆盖value,没有链表尾部追加节点(利用 synchronized 锁写入数据)
6.如果这个节点的类型是TreeBin的话,直接调用红黑树的插入方法进行插入新的节点
7.插入完节点之后再次检查链表长度 ,如果长度大于8,就把这个链表转换成红黑树;
8.插入完后对当前容量大小进行检查 ,如果超过了临界值(实际大小*加载因子)就需要扩容。
5.hashmap的实现原理说一下
HashEntry数组+链表+红黑树。key的HashCode定位到数组,无链表则一次寻址找到。有链表为了解决哈希冲突…
6.hashmap的加载因子有什么用,加载因子为0或1会怎样
加载因子是表示Hsah表中元素的填满的程度.若:加载因子越大,填满的元素越多,好处是,空间利用率高了,但:冲突的机会加大了.反之,加载因子越小,填满的元素越少,好处是:冲突的机会减小了,但:空间浪费多了.尽量减少哈希冲突。为0浪费内存空间,加载因子范围0.1-1,为1空间利用最大化,哈希碰撞频繁。效率低下。
7.hashmap的初始容量为什么是16,如果初始容量为15或17会怎样,哪个初始容量的hashmap性能好
16更好,长度16或者其他2的幂,Length-1的值为奇数,二进制的最后一位一定是1,这样便保证了 h&(length-1) 的最后一位可能为0,也可能为1(这取决于h的值),即与后的结果可能为偶数,也可能为奇数,这样便可以保证散列的均匀性。用16,是为了得到的15二进制是1111。
而如果 length 为奇数的话,很明显 length-1 为偶数,它的最后一位是0,这样 h&(length-1) 的最后一位肯定为0,即只能为偶数,这样任何hash值都只会被散列到数组的偶数下标位置上,这便浪费了近一半的空间 ,因此,length 取2的整数次幂,是为了使不同 hash 值发生碰撞的概率较小,这样就能使元素在哈希表中均匀地散列。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值