如何实现线程安全的HashMap

一、为什么HashMap线程不安全?

(1)内部存储结构:HashMap内部存储使用了一个Node数组(默认大小是16),如果存在相同的hashcode和相同的key的元素,那么新值覆盖原来的旧值;

如果存在相同的hashcode,那么他们的索引位置就相同,这时判断他们的key是否相同,如果不相同,这时就是产生了hash冲突,这时数据放在一个 Entry 链。 

(2)自动扩容机制:如果多个线程同时检测到元素的个数超过阀值(数组大小*负载因子),多个进程会同时对Node数组进行扩容,都在重新计算元素位置以及复制数据,但是最终只有一个线程扩容后的数组会赋给table,其他线程的都会丢失,并且各自线程put的数据也丢失。

二、 如何线程安全的使用HashMap?

1.HashTable

Map<String, String> hashtable = new Hashtable<>();

HashTable使用synchronized来保证线程安全的,所有线程竞争同一把锁,效率低

2.ConcurrentHashMap 

Map<String, String> concurrentHashMap = new ConcurrentHashMap<>();

使用锁分段技术:它包含一个segment数组,将数据分段存储,给每一段数据配一把锁,效率高

Java8中使用CAS算法

3.Synchronized Map

Map<String, String> synchronizedHashMap = Collections.synchronizedMap(new HashMap<String, String>());

调用synchronizedMap()方法后会返回一个SynchronizedMap类的对象,而在SynchronizedMap类中使用了synchronized同步关键字来保证对Map的操作是安全的。
————————————————
版权声明:本文为CSDN博主「Michaeles」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/Michaeles/article/details/86178918

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值