Java基础 -> ConcurrentHashMap的实现原理

ConcurrentHashMap的实现原理

在并发编程中使用HashMap可能导致程序死循环。而使用线程安全的HashTable效率又非常低下,基于以上两个原因,便有了ConcurrentHashMap的登场机会

ConcurrentHashMap所使用的锁分段技术。

  1. 首先将数据分成一段一段地存储
  2. 然后给每一段数据配一把锁,当一个线程占用锁访问其中一个段数据的时候,其他段的数据也能被其他线程访问。

jdk1.7

了解了解就行,现在jdk大多都是1.8,几乎重写了一边

采用Segment + HashEntry的方式进行实现,结构如下:

img

  • ConcurrentHashMap是由Segment数组结构HashEntry数组结构(类似Map)组成。

  • Segment是一种可重入锁(ReentrantLock),在ConcurrentHashMap里扮演锁的角色;HashEntry则用于存储键值对数据。

  • 一个ConcurrentHashMap里包含一个Segment数组。

  • Segment的结构和HashMap类似,是一种数组和链表结构。

  • Segment数组的每个节点都相当于一个小HashMap,这样就可以对单个Segment节点加锁,不会影响到所有数据都被锁上

  • 原理跟HashMap差不多,都是用hash算法找位置

    • 第一次hash找Segment数组的某个节点
    • 第二次hash找这个Segment节点下这个小HashMap的数组位置(就类似HashMap的操作了)

jdk1.8

ConcurrentHashMap在1.8中的实现,相比于1.7的版本基本上全部都变掉了。

  1. 首先,取消了Segment分段锁的数据结构,取而代之的是数组+链表(红黑树)的结构(回到了HashMap结构)。
  2. 而对于锁的粒度,调整为对每个数组元素加锁(就是对每个链表或红黑树加锁)
  3. ConcurrentHashMap采用 CAS + Synchronized来保证并发安全进行实现(乐观锁+悲观锁)

img

ConcurrentHashMap的put实现

  • 与HashMap稍微不同就是,HashMap会计算hash和equals()比较后直接加上去

  • ConcurrentHashMap会有相应锁的控制(CAS+synchronized)后再添加数据

    • 执行第一次put方法时,采用CAS机制
    • 链表or树的其他节点put就是用synchronized加锁,利用头节点锁住整个链表或者树结构
    • 后面跟HashMap差不多了

    为什么不都用CAS机制,而是有一部分用synchronized呢?

    • 还是因为CAS的缺点,不能对多个变量进行原子操作,不能保证代码块的原子性
    • 并且CAS的预测值与实际值比较并不包括next的属性,仅仅对(k,v)进行比较
    • 所以对同一节点进行操作时
      • 线程1要修改value属性,线程2修改next属性
      • 线程2执行快,先替换了next属性
      • 而线程1预测(k,v)并没有变化,又将之前的旧next覆盖掉线程2修改后的

    大帅比,点赞了没?

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值