区别:

  • LinkedHashMap 是继承于 HashMap,是基于 HashMap 和双向链表来实现的。
  • HashMap 无序;LinkedHashMap 有序,可分为插入顺序和访问顺序两种。如果是访问顺序,那 put 和 get 操作已存在的 Entry 时,都会把 Entry 移动到双向链表的表尾 (其实是先删除再插入)。
  • LinkedHashMap 存取数据,还是跟 HashMap 一样使用的 Entry[] 的方式,双向链表只是为了保证顺序。
  • LinkedHashMap 是线程不安全的。

LinkedHashMap 应用场景

HashMap 是无序的,当我们希望有顺序地去存储 key-value 时,就需要使用 LinkedHashMap 了。

  1. Map<String, String> hashMap = new HashMap<String, String>();
  2. hashMap.put("name1", "josan1");
  3. hashMap.put("name2", "josan2");
  4. hashMap.put("name3", "josan3");
  5. Set<Entry<String, String>> set = hashMap.entrySet();
  6. Iterator<Entry<String, String>> iterator = set.iterator();
  7. while(iterator.hasNext()) {
  8. Entry entry = iterator.next();
  9. String key = (String) entry.getKey();
  10. String value = (String) entry.getValue();
  11. System.out.println("key:" + key + ",value:" + value);
  12. }

LinkedHashMap和hashMap和TreeMap的区别 - 图1

我们是按照 xxx1、xxx2、xxx3 的顺序插入的,但是输出结果并不是按照顺序的。

同样的数据,我们再试试 LinkedHashMap

  1. Map<String, String> linkedHashMap = new LinkedHashMap<>();
  2. linkedHashMap.put("name1", "josan1");
  3. linkedHashMap.put("name2", "josan2");
  4. linkedHashMap.put("name3", "josan3");
  5. Set<Entry<String, String>> set = linkedHashMap.entrySet();
  6. Iterator<Entry<String, String>> iterator = set.iterator();
  7. while(iterator.hasNext()) {
  8. Entry entry = iterator.next();
  9. String key = (String) entry.getKey();
  10. String value = (String) entry.getValue();
  11. System.out.println("key:" + key + ",value:" + value);
  12. }

LinkedHashMap和hashMap和TreeMap的区别 - 图2

结果可知,LinkedHashMap 是有序的,且默认为插入顺序。

插入顺序和访问顺序

LinkedHashMap 默认的构造参数是默认 插入顺序的,就是说你插入的是什么顺序,读出来的就是什么顺序,但是也有访问顺序,就是说你访问了一个 key,这个 key 就跑到了最后面。

这里 accessOrder 设置为 false,表示不是访问顺序而是插入顺序存储的,这也是默认值,表示 LinkedHashMap 中存储的顺序是按照调用 put 方法插入的顺序进行排序的。LinkedHashMap 也提供了可以设置 accessOrder 的构造方法,我们来看看这种模式下,它的顺序有什么特点?

  1. // 第三个参数用于指定accessOrder值
  2. Map<String, String> linkedHashMap = new LinkedHashMap<>(16, 0.75f, true);
  3. linkedHashMap.put("name1", "josan1");
  4. linkedHashMap.put("name2", "josan2");
  5. linkedHashMap.put("name3", "josan3");
  6. System.out.println("开始时顺序:");
  7. Set<Entry<String, String>> set = linkedHashMap.entrySet();
  8. Iterator<Entry<String, String>> iterator = set.iterator();
  9. while(iterator.hasNext()) {
  10. Entry entry = iterator.next();
  11. String key = (String) entry.getKey();
  12. String value = (String) entry.getValue();
  13. System.out.println("key:" + key + ",value:" + value);
  14. }
  15. System.out.println("通过get方法,导致key为name1对应的Entry到表尾");
  16. linkedHashMap.get("name1");
  17. Set<Entry<String, String>> set2 = linkedHashMap.entrySet();
  18. Iterator<Entry<String, String>> iterator2 = set2.iterator();
  19. while(iterator2.hasNext()) {
  20. Entry entry = iterator2.next();
  21. String key = (String) entry.getKey();
  22. String value = (String) entry.getValue();
  23. System.out.println("key:" + key + ",value:" + value);
  24. }

LinkedHashMap和hashMap和TreeMap的区别 - 图3

因为调用了 get(“name1”) 导致了 name1 对应的 Entry 移动到了最后,这里只要知道 LinkedHashMap 有插入顺序和访问顺序两种就可以。

TreeMap 的用法(主要是排序)

TreeMap中默认的排序为升序,如果要改变其排序可以自己写一个Comparator。

  1. import java.util.Comparator;
  2. import java.util.Iterator;
  3. import java.util.Set;
  4. import java.util.TreeMap;
  5. public class Compare {
  6. public static void main(String[] args) {
  7. TreeMap<String,Integer> map = new TreeMap<String,Integer>(new xbComparator());
  8. map.put("key_1", 1);
  9. map.put("key_2", 2);
  10. map.put("key_3", 3);
  11. Set<String> keys = map.keySet();
  12. Iterator<String> iter = keys.iterator();
  13. while(iter.hasNext())
  14. {
  15. String key = iter.next();
  16. System.out.println(" "+key+":"+map.get(key));
  17. }
  18. }
  19. }
  20. class xbComparator implements Comparator
  21. {
  22. public int compare(Object o1,Object o2)
  23. {
  24. String i1=(String)o1;
  25. String i2=(String)o2;
  26. return -i1.compareTo(i2);
  27. }
  28. }

LinkedHashMap和hashMap和TreeMap的区别 - 图4