HashMap、Hashtable、LinkedHashMap、TreeMap、ConcurrentHashMap的区别
Map是Java最常用的集合类之一。它有很多实现类,我总结了几种常用的Map实现类,如下图所示。本篇文章重点总结几个Map实现类的特点和区别:
特点总结:
实现类 |
HashMap |
LinkedHashMap |
TreeMap |
Hashtable |
ConcurrentHashMap |
允许key/value为空 |
允许为空 |
允许为空 |
允许为空 |
不允许为空 |
不允许为空 |
线程安全 |
不安全 |
不安全 |
不安全 |
安全(全部锁) |
安全(部分锁) |
排序 |
无序 |
迭代器顺序 |
红黑树顺序(可以自定义) |
无序 |
无序 |
1、HashMap: Map最常用的实现类
a、HashMap可以存空的key和空的value。
public void nullKeyTest() {
Map<String, Object> map = new HashMap<String, Object>();
map.put(null, "value");
System.out.println(map); // {null=value}
}
public void nullValueTest() {
Map<String, Object> map = new HashMap<String, Object>();
map.put("key", null);
System.out.println(map); // {key=null}
}
b、HashMap线程不安全,在多线程的情况下,需要额外方法实现数据的同步。Collections提供了同步的方法,如下:返回了一个线程安全的HashMap类。(PS:这时候可以考虑ConcurrentHashMap,下文有介绍)。
public void synchMapTest() {
Map<String,Object> map = Collections.synchronizedMap(new HashMap<String,Object>());
// returned a synchoronized HashMap
}
2、LinkedHashMap: HashMap子类
a、LinkedHashMap作为HashMap的子类,同样也可以存空的key和空的value。
b、LinkedHashMap会按存入顺序存放元素(迭代器顺序)。
现实中很多情况下都可以使用LinkedHashMap,例如购物车的实现:按用户添加的时间顺序显示购物车中的元素。
以下map的输出是有顺序的,而map1是无序的。
public void iterateOrderTest() {
Map<String, Object> map = new LinkedHashMap<String, Object>();
map.put("ccc", "value");
map.put("bbb", "value");
map.put("aaa", "value");
System.out.println(map); // {ccc=value, bbb=value, aaa=value} Map<String, Object> map1 = new HashMap<String, Object>();
map1.put("ccc", "value");
map1.put("bbb", "value");
map1.put("aaa", "value");
System.out.println(map1); // {aaa=value, ccc=value, bbb=value}
}
c、LinkedHashMap有个方法叫:removeEldestEntry(Map.Entry) ,用户可以重写该方法来制定添加元素时的策略。方法内部返回true时,当要往该map中存放(put\putAll)元素时,就先删除最老的对象,然后再添加新对象。使用实例:在设计Cache(缓存)时,可以重写该方法来制定键值的过期规则。
API:http://docs.oracle.com/javase/6/docs/api/java/util/LinkedHashMap.html#removeEldestEntry(java.util.Map.Entry)
例子:在新建一个LinkedHashMap对象时,重写了其removeEldestEntry方法,如果map的容易大于2时,就返回true,于是最老的键aaa就被删除了。
public void removeEldestEntryTest() {
Map<String, Object> map = new LinkedHashMap<String, Object>() {
private int MAX_ENTRIES = 2;
protected boolean removeEldestEntry(Map.Entry<String, Object> eldest) {
return size() > MAX_ENTRIES;
}
}; map.put("aaa", "value");
map.put("bbb", "value");
map.put("ccc", "value");
System.out.println(map); // {bbb=value, ccc=value}
}
所以如果不是为了键值的顺序,还是使用HashMap最好,因为LinkedHashMap需要额外的空间去维护键值之间的链表关系。
3、Hashtable类
Hashtable是JDK1.0就有的,算是很古老的Map实现了,前辈们设计的初衷就是为了线程安全考虑的(有点类似List里的一个实现类,叫Vector)。
另外就是Hashtable不能存放空的key/value,会报java.lang.NullPointerException错误。
值得一提的是,现在很少会用到Hashtable类,如果要在多线程的情况下使用线程安全类,那会考虑ConcurrentHashMap,下文有介绍。
4、SortedMap的实现类:TreeMap
TreeMap红黑树的一个实现。其键值按照自然顺序排序的。它的构造函数中提供了参数Comparator,可以自定义实现排序。
另外,TreeMap也是线程不安全的。
构造器API:
TreeMap(Comparator<? super K> comparator)
// 返回一个空的tree map,按传入的comparator规则进行排序。
排序的例子:
public void orderTest() {
Map<String, String> treeMap = new TreeMap<String, String>();
treeMap.put("aaa", "value1");
treeMap.put("bbb", "value2");
treeMap.put("ccc", "value3");
System.out.println(treeMap); // {aaa=value1, bbb=value2, ccc=value3}
} // 自定义排序器:按键值的倒序排序:
public void comparatorTest() {
Map<String,String> treeMap1 = new TreeMap<String,String>(new Comparator<String>(){
public int compare(String o1, String o2) {
return -o1.compareTo(o2);
}
});
treeMap1.put("aaa", "value1");
treeMap1.put("bbb", "value2");
treeMap1.put("ccc", "value3");
System.out.println(treeMap1); // {ccc=value3, bbb=value2, aaa=value1}
}
5、ConcurrentMap的实现类:ConcurrentHashMap
ConcurrentHashMap是1.5版本引进的。Concurrent的意思是同时发生的,所以这个实现类的重点是线程安全。(有点类似于Hashtable,但也有区别)。
ConcurrentHashMap的默认并发级别是16。
ConcurrentHashMap的设计初衷是优化线程安全中的读操作,ConcurrentHashMap在取元素的操作(包括get)通常没有加锁,而对于另外一个线程安全的Hashtable来说,是所有方法都加同步锁的。
一般来说,在JAVA1.5版本以后,推荐使用的是ConcurrentHashMap,而Hashtable有点像是历史遗留的类而已。
以下是国外的一篇文章中摘录的ConcurrentHashMap和Hashtable的性能对比:
Threads |
ConcurrentHashMap |
Hashtable |
1 |
1.00 |
1.03 |
2 |
2.59 |
32.40 |
4 |
5.58 |
78.23 |
8 |
13.21 |
163.48 |
16 |
27.58 |
341.21 |
32 |
57.27 |
778.41 |
HashMap、Hashtable、LinkedHashMap、TreeMap、ConcurrentHashMap的区别的更多相关文章
- 接口java.util.Map的四个实现类HashMap Hashtable LinkedHashMap TreeMap
java中HashMap,LinkedHashMap,TreeMap,HashTable的区别 :java为数据结构中的映射定义了一个接口java.util.Map;它有四个实现类,分别是HashMa ...
- HashMap,HashTable ,LinkedHashMap,TreeMap的区别
Map:主要是存储键值对,不允许键重复,但可以值重复. HashMap:根据键的HashCode值来存储数据,根据键直接获取值.具有很快的访问速度,遍历时,取得的数据值的顺序都是随机的.hashMap ...
- HashMap Hashtable LinkedHashMap TreeMap
// Map<String, String> map = new HashMap<String, String>(); // bb aa cc Map<String, S ...
- HashMap,LinkedHashMap,TreeMap之间的区别
java为数据结构中的映射定义了一个接口java.util.Map;它有四个实现类,分别是HashMap Hashtable LinkedHashMap 和TreeMap . Map 主要用于存储键( ...
- HashMap Hashtable LinkedHashMap 和TreeMap
Map主要用于存储健值对,根据键得到值,因此不允许键重复(重复了覆盖了),但允许值重复.Hashmap 是一个最常用的Map,它根据键的HashCode值存储数据,根据键可以直接获取它的值,具有很快的 ...
- HashSet、HashMap、Hashtable、TreeMap循环、区别
HashSet 循环 //可以为null HashSet<Object> hashSet =new HashSet<Object>(); hashSet.add(1); has ...
- Java HashMap、HashTable、TreeMap、WeakHashMap区别
1.HashMap不是线程安全,而HashTable是线程安全
- HashMap、LinkedHashMap、ConcurrentHashMap、ArrayList、LinkedList 底层实现
HashMap相关问题 1.你用过HashMap吗?什么是HashMap?你为什么用到它? 用过,HashMap是基于哈希表的Map接口的非同步实现,它允许null键和null值,且HashMap依托 ...
- HashMap、LinkedHashMap、ConcurrentHashMap、ArrayList、LinkedList的底层实现
HashMap:底层是一个数组+链表实现 LinkedHashMap:底层是Hash表和链表的实现 ConcurrentHashMap:基于双数组和链表的Map接口的同步实现 ArrayList:底层 ...
- 集合类源码(六)Map(HashMap, Hashtable, LinkedHashMap, WeakHashMap)
HashMap 内部结构 内部是一个Node数组,每个Node都是链表的头,当链表的大小达到8之后链表转变成红黑树. put操作 final V putVal(int hash, K key, V v ...
随机推荐
- vue-quill-editor 封装成组件;图片文件流上传;同一页面多个编辑器样式异常解决办法
使用方法: 引入并注册组件,然后直接使用: @getcode是同步获取编辑器内容的::contentDefault是编辑器的默认内容: 注意:如果同一个页面多个编辑器,参数id不能相同,否则只有第一个 ...
- 规模化落地云原生,阿里云即将重磅亮相 KubeCon China
2019 年 6 月 24 日至 26 日, 由 Cloud Native Computing Foundation (CNCF) 主办的云原生技术大会 KubeCon + CloudNativeCo ...
- tensorflow学习笔记(三十四):Saver(保存与加载模型)
Savertensorflow 中的 Saver 对象是用于 参数保存和恢复的.如何使用呢? 这里介绍了一些基本的用法. 官网中给出了这么一个例子: v1 = tf.Variable(..., nam ...
- Oracle基础学习4--Oracle权限传递
版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/wang379275614/article/details/32215325 以下将用一个实例来解说: ...
- 检查进程启动情况,开始时间、启动时间、启动进程数、进程数是否正确、PID
#!/bin/sh bin=$(cd ``;pwd) cd ${bin} ### 定义检查函数 chk(){ programName=$ correctNum=$ programSubName=$ # ...
- 利用scrapy爬取文件后并基于管道化的持久化存储
我们在pycharm上爬取 首先我们可以在本文件打开命令框或在Terminal下创建 scrapy startproject xiaohuaPro ------------创建文件 scrapy ...
- hdu 1430 魔板 (BFS+预处理)
Problem - 1430 跟八数码相似的一题搜索题.做法可以是双向BFS或者预处理从"12345678"开始可以到达的所有状态,然后等价转换过去直接回溯路径即可. 代码如下: ...
- [C#] 调试silverlight的时候,总是报“向占位程序传送了空的索引指针”
这是由于visual studio在调试silverlight的时候,必须和ie一起工作. 按照以下步骤可以把ie设为visual studio的默认浏览器(不用修改操作系统的默认浏览器): 1) 在 ...
- vscode编辑如何保存时自动校准eslint规范
在日常开发中,一个大点的项目会有多人参与,那么可能就会出现大家的代码风格不一,各显神通,这个时候就要祭出我们的eslint. 在这之前磨刀不误砍柴工,我们先来配置一下我们的代码编辑工具,如何在vsco ...
- HDU 3397"Sequence operation"(线段树区间和并)
传送门 •题意 给你一个仅包含 0 和 1 的序列: 在这个序列上有如下操作: (1)0 a b : 将 [a,b] 区间的数置 0: (2)1 a b : 将 [a,b] 区间的数置 1: (3)2 ...