hashmap在java哪个包_Java并发包之ConcurrentHashMap

本文分析并补充一下线程安全的HashMap-ConcurrentHashMap

参考优秀博文: 深入并发包 ConcurrentHashMap

补充:

CocurrentHashMap能完全代替Hashtable吗?

我们知道Hashtable是synchronized的,会锁定整个map,当Hashtable的大小增加到一定的时候,性能会急剧下降,因为迭代时需要被锁定很长的时间。ConcurrentHashMap同步性能更好,因为它仅仅根据同步级别对map的一部分进行上锁。

HashTable虽然性能上不如ConcurrentHashMap,但并不能完全被取代,两者的迭代器的一致性不同的,HashTable的迭代器是强一致性的,而ConcurrentHashMap是弱一致的。 ConcurrentHashMap的get,clear,iterator 都是弱一致性的。 Doug Lea 也将这个判断留给用户自己决定是否使用ConcurrentHashMap。

那么什么是强一致性和弱一致性呢?

get方法是弱一致的,是什么含义?可能你期望往ConcurrentHashMap底层数据结构中加入一个元素后,立马能对get可见,但ConcurrentHashMap并不能如你所愿。换句话说,put操作将一个元素加入到底层数据结构后,get可能在某段时间内还看不到这个元素,若不考虑内存模型,单从代码逻辑上来看,却是应该可以看得到的。

ConcurrentHashMap.computeIfAbsent 在 JDK 1.8 存在 BUG

public class ConcurrentHashMapDemo{

private Map cache =new ConcurrentHashMap<>(15);

public static void main(String[]args){

ConcurrentHashMapDemo ch = new ConcurrentHashMapDemo();

System.out.println(ch.fibonaacci(80));

}

public int fibonaacci(Integer i){

if(i==0||i ==1) {

return i;

}

return cache.computeIfAbsent(i,(key) -> {

System.out.println("fibonaacci : "+key);

return fibonaacci(key -1)+fibonaacci(key - 2);

});

}

}

如果你将这个代码跑起来,你会发现的这个程序将进入死循环,而无法结束。通过阅读源码发现,ConcurrentHashMap.computeIfAbsent() 方法停留在了一个 ReservationNode 对象上,感兴趣的朋友可阅读java.util.concurrent.ConcurrentHashMap#transfer扩容源码

解决方法:

升级到JDK 1.9,已经修复该Bug

不要用递归的方式创建 ConcurrentHashMap 对象

技术讨论 & 疑问建议 & 个人博客

版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 3.0 许可协议,转载请注明出处!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
线程安全的HashMap通常使用锁或者并发数据结构来实现。以下是两种常见的实现方式: 1. 使用锁实现线程安全的HashMap 在单线程环境下,HashMap是非常高效的数据结构。但是在多线程环境下,由于线程之间的竞争,可能会导致HashMap出现错误或者不一致的状态。为了解决这个问题,可以使用锁来保证线程安全。常见的实现方式括: - 使用synchronized关键字来保护HashMap的读写操作。这种方式比较简单,但是会导致性能下降,因为每个线程都需要获得锁才能进行读写操作。 - 使用ConcurrentHashMap代替HashMapConcurrentHashMapJava并发提供的高性能线程安全Map实现。它利用分段锁来实现线程安全,可以同时支持多个读操作和少量写操作,因此在高并发环境下性能比较好。 2. 使用并发数据结构实现线程安全的HashMap 除了使用锁,还可以使用Java并发提供的一些并发数据结构来实现线程安全的HashMap。例如: - ConcurrentHashMap。这是Java并发提供的高性能线程安全Map实现,它利用分段锁来实现线程安全,可以同时支持多个读操作和少量写操作,因此在高并发环境下性能比较好。 - ConcurrentSkipListMap。这是Java并发提供的高性能线程安全有序Map实现,它使用跳表来实现数据结构,可以支持快速的查找、插入和删除操作,而且可以保证数据的有序性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值