我们说ConcurrentHashMap是用在多并发环境下的HashMap
JDK1.7
segment数组+HashEntry数组+链表(即两个数组+挂一个链表)
上锁的时候锁的是segment数组,采用分段锁,每一把锁只锁一个Segment,默认是这个数组大小为16,所以最多可以有16个线程进行并发操作
segemnt数组中的元素下面对应的是HashEntry数组(这个数组是可以扩容的)
JDK1.8的ConcurrentHashMap已经和HashMap的底层数据结构很相似了
当数组的长度扩容到64了,而且链表长度达到8之后,再往链表中放元素就会将链表变成红黑树,
如何保证线程安全:
初始化Node数组采用CAS+volatile
放数据的时候采用通过:synchronized
1.HashMap没有考虑同步,是线程不安全的;
Hashtable使用了synchronized关键字,是线程安全的;
HashMap允许K/V都为null;后者K/V都不允许为null;
2.HashMap底层与HashTable原理相同,Java 8版本以后,链表长度大于8就会变成红黑树。
3.HashMap没有考虑同步,HashTable考虑了同步的问题
但是 HashTable 在每次同步执行时都要锁住整个结构。
ConcurrentHashMap 锁的方式是稍微细粒度的,只锁segment数组中的一个segment