Map集合体系


java map key有两个 两个key的map_线程安全

Map

Map是一种双列集合

它的每个元素都包含一个键对象key和一个值对象value

键和值之间存在一种对应关系,称为映射

从Map集合中访问元素,只要指定了key,就能找到对应的value。

HashMap

(数组+链表/红黑树)

HashMap实现原理

HashMap是基于Hash算法实现的,通过put(kty,value)存储,get(kty,value)获取。

当传入key是,HashMap会根据key.hashCode()计算出hash值,根据hash值将value保存在bucket中。

当计算出的hash值相同时,称为哈希冲突。

HashMap的做法是用链表和红黑树(jdk1.8)存储相同的hash值的value。

当hash冲突个数较少,使用链表,否则使用红黑树。

特点:

HashMap去掉了contains(),改成containsKey()和containsValue()

非线程安全,允许键值为null,存取顺序不一致。

HaspMap允许键值为空

默认大小16,负载因子大小为0.75

取出Map中的键值该如何遍历

方式一:获取键的集合、遍历键的集合、获取每个键对应的值

方式二:获取集合中映射关系、获取映射关系中的键、获取映射中的值

LinkedHashMap

(数组+红黑数+双向链表)

继承自HashMap是 HashMap 的一个子类,允许key和value都为空,保存了记录的插入顺序,在用 Iterator 遍历 LinkedHashMap时,先得到的记录肯定是先插入的,也可以在构造时带参数,按照访问次序(有序)排序。

ConcurrentHashMap

(分段数组+链表/红黑树)

Java5提供了ConcurrentHashMap,它是HashTable的替代

1.7与1.8区别

jdk1.7中采用ReentrantLock+segment+HashEntry的方式,lock加在Segment上。Size计算采用不加锁的方式,连续计算元素的个数,最多计算3次,如果前后两次计算结果相同,说明元素个数准确,如果前后两次不同,则给segment加锁,在计算一次。通过segment分段锁保证线程安全。

jdk1.8舍弃了segment,取代的是CAS+synchronized+HashEntry+红黑树的方式,使用volatile类型的变量baseCount记录元素的个数,当插入新数据或删除数据时,通过addCount()方法更新baseCount,通过累加baseCount和CounterCell数组中的数量得到元素总数。通过CAS+synchronized保证线程安全。

实现原理

ConcurrentHashMap类中包含两个静态内部类HashEntry和segment。HashEntry用来封装映射表的键值对。Segment用来充当锁的角色,每个Segment对象守护整个散列表的若干个桶。每个桶由若干个HashEntry对象连接起来的链表。

(1) Segment段

ConcurrentHashMap 和 HashMap 思路是差不多的,但是因为它支持并发操作,所以要复杂一 些。整个 ConcurrentHashMap 由一个个 Segment 组成,Segment 代表”部分“或”一段“的 意思,所以很多地方都会将其描述为分段锁。注意,行文中,我很多地方用了“槽”来代表一个 segment。

(2)线程安全(segment继承ReentrantLocak加锁)

简单理解就是,ConcurrentHashMap 是一个 Segment 数组,Segment 通过继承 ReentrantLock 来进行加锁,所以每次需要加锁的操作锁住的是一个 segment,这样只要保证每 个 Segment 是线程安全的,也就实现了全局的线程安全。

HashTable

Hashtable 是遗留类,很多映射的常用功能与 HashMap 类似,不同的是它承自 Dictionary 类,并且是线程安全的。

任一时间只有一个线程能写 Hashtable,并发性不如 ConcurrentHashMap, 因为 ConcurrentHashMap 引入了分段锁。

Hashtable 不建议在新代码中使用,不需要线程安全的场合可以用HashMap替换,需要线程安全的场合可以用ConcurrentHashMap替换。

Properties

是Hashtable的子类  

读写资源配置文件  

键与值都只能字符串

p.setProperties(“key”,”value”)  
p.getProperties(“key”,”value”)

TreeMap

TreeMap 实现 SortedMap 接口,能够把它保存的记录根据键排序,默认是按键值的升序排序, 也可以指定排序的比较器,当用Iterator遍历TreeMap时,得到的记录是排过序的。

如果使用排序的映射,建议使用TreeMap。

在使用 TreeMap 时,key 必须实现 Comparable 接口或者在构造 TreeMap 传入自定义的 Comparator,否则会在运行时抛出java.lang.ClassCastException类型的异常。