一.TreeMap介绍

TreeMap 简介

TreeMap 是一个有序的key-value集合,它是通过红黑树实现的。
TreeMap 继承于AbstractMap,所以它是一个Map,即一个key-value集合。
TreeMap 实现了NavigableMap接口,意味着它支持一系列的导航方法。比如返回有序的key集合。
TreeMap 实现了Cloneable接口,意味着它能被克隆
TreeMap 实现了java.io.Serializable接口,意味着它支持序列化

TreeMap基于红黑树(Red-Black tree)实现。该映射根据其键的自然顺序进行排序,或者根据创建映射时提供的 Comparator 进行排序,具体取决于使用的构造方法。
TreeMap的基本操作 containsKey、get、put 和 remove 的时间复杂度是 log(n) 。
另外,TreeMap是非同步的。 它的iterator 方法返回的迭代器是fail-fastl的。

TreeMap的构造函数

  1. // 默认构造函数。使用该构造函数,TreeMap中的元素按照自然排序进行排列。
  2. TreeMap()
  3. // 创建的TreeMap包含Map
  4. TreeMap(Map<? extends K, ? extends V> copyFrom)
  5. // 指定Tree的比较器
  6. TreeMap(Comparator<? super K> comparator)
  7. // 创建的TreeSet包含copyFrom
  8. TreeMap(SortedMap<K, ? extends V> copyFrom)

TreeMap的API

  1. Entry<K, V> ceilingEntry(K key)
  2. K ceilingKey(K key)
  3. void clear()
  4. Object clone()
  5. Comparator<? super K> comparator()
  6. boolean containsKey(Object key)
  7. NavigableSet<K> descendingKeySet()
  8. NavigableMap<K, V> descendingMap()
  9. Set<Entry<K, V>> entrySet()
  10. Entry<K, V> firstEntry()
  11. K firstKey()
  12. Entry<K, V> floorEntry(K key)
  13. K floorKey(K key)
  14. V get(Object key)
  15. NavigableMap<K, V> headMap(K to, boolean inclusive)
  16. SortedMap<K, V> headMap(K toExclusive)
  17. Entry<K, V> higherEntry(K key)
  18. K higherKey(K key)
  19. boolean isEmpty()
  20. Set<K> keySet()
  21. Entry<K, V> lastEntry()
  22. K lastKey()
  23. Entry<K, V> lowerEntry(K key)
  24. K lowerKey(K key)
  25. NavigableSet<K> navigableKeySet()
  26. Entry<K, V> pollFirstEntry()
  27. Entry<K, V> pollLastEntry()
  28. V put(K key, V value)
  29. V remove(Object key)
  30. int size()
  31. SortedMap<K, V> subMap(K fromInclusive, K toExclusive)
  32. NavigableMap<K, V> subMap(K from, boolean fromInclusive, K to, boolean toInclusive)
  33. NavigableMap<K, V> tailMap(K from, boolean inclusive)
  34. SortedMap<K, V> tailMap(K fromInclusive)

二.TreeMap数据结构

TreeMap的继承关系

  1. java.lang.Object
  2. java.util.AbstractMap<K, V>
  3. java.util.TreeMap<K, V>
  4. public class TreeMap<K,V>
  5. extends AbstractMap<K,V>
  6. implements NavigableMap<K,V>, Cloneable, java.io.Serializable {}

TreeMap与Map关系如下图:

TreeMap详细介绍和使用示例 - 图1

从图中可以看出:
(01) TreeMap实现继承于AbstractMap,并且实现了NavigableMap接口。
(02) TreeMap的本质是R-B Tree(红黑树),它包含几个重要的成员变量: root, size, comparator。
  root 是红黑数的根节点。它是Entry类型,Entry是红黑数的节点,它包含了红黑数的6个基本组成成分:key(键)、value(值)、left(左孩子)、right(右孩子)、parent(父节点)、color(颜色)。Entry节点根据key进行排序,Entry节点包含的内容为value。
  红黑数排序时,根据Entry中的key进行排序;Entry中的key比较大小是根据比较器comparator来进行判断的。
  size是红黑数中节点的个数。

关于红黑数的具体算法,请参考”红黑树(一) 原理和算法详细介绍“。

四.TreeMap遍历方式

4.1 遍历TreeMap的键值对

第一步:根据entrySet()获取TreeMap的“键值对”的Set集合。
第二步:通过Iterator迭代器遍历“第一步”得到的集合。

  1. // 假设map是TreeMap对象
  2. // map中的key是String类型,value是Integer类型
  3. Integer integ = null;
  4. Iterator iter = map.entrySet().iterator();
  5. while(iter.hasNext()) {
  6. Map.Entry entry = (Map.Entry)iter.next();
  7. // 获取key
  8. key = (String)entry.getKey();
  9. // 获取value
  10. integ = (Integer)entry.getValue();
  11. }

4.2 遍历TreeMap的键

第一步:根据keySet()获取TreeMap的“键”的Set集合。
第二步:通过Iterator迭代器遍历“第一步”得到的集合。

  1. // 假设map是TreeMap对象
  2. // map中的key是String类型,value是Integer类型
  3. String key = null;
  4. Integer integ = null;
  5. Iterator iter = map.keySet().iterator();
  6. while (iter.hasNext()) {
  7. // 获取key
  8. key = (String)iter.next();
  9. // 根据key,获取value
  10. integ = (Integer)map.get(key);
  11. }

4.3 遍历TreeMap的值

第一步:根据value()获取TreeMap的“值”的集合。
第二步:通过Iterator迭代器遍历“第一步”得到的集合。

  1. // 假设map是TreeMap对象
  2. // map中的key是String类型,value是Integer类型
  3. Integer value = null;
  4. Collection c = map.values();
  5. Iterator iter= c.iterator();
  6. while (iter.hasNext()) {
  7. value = (Integer)iter.next();
  8. }

TreeMap遍历测试程序如下:

  1. import java.util.Map;
  2. import java.util.Random;
  3. import java.util.Iterator;
  4. import java.util.TreeMap;
  5. import java.util.HashSet;
  6. import java.util.Map.Entry;
  7. import java.util.Collection;
  8. /*
  9. * @desc 遍历TreeMap的测试程序。
  10. * (01) 通过entrySet()去遍历key、value,参考实现函数:
  11. * iteratorTreeMapByEntryset()
  12. * (02) 通过keySet()去遍历key、value,参考实现函数:
  13. * iteratorTreeMapByKeyset()
  14. * (03) 通过values()去遍历value,参考实现函数:
  15. * iteratorTreeMapJustValues()
  16. *
  17. * @author skywang
  18. */
  19. public class TreeMapIteratorTest {
  20. public static void main(String[] args) {
  21. int val = 0;
  22. String key = null;
  23. Integer value = null;
  24. Random r = new Random();
  25. TreeMap map = new TreeMap();
  26. for (int i=0; i<12; i++) {
  27. // 随机获取一个[0,100)之间的数字
  28. val = r.nextInt(100);
  29. key = String.valueOf(val);
  30. value = r.nextInt(5);
  31. // 添加到TreeMap中
  32. map.put(key, value);
  33. System.out.println(" key:"+key+" value:"+value);
  34. }
  35. // 通过entrySet()遍历TreeMap的key-value
  36. iteratorTreeMapByEntryset(map) ;
  37. // 通过keySet()遍历TreeMap的key-value
  38. iteratorTreeMapByKeyset(map) ;
  39. // 单单遍历TreeMap的value
  40. iteratorTreeMapJustValues(map);
  41. }
  42. /*
  43. * 通过entry set遍历TreeMap
  44. * 效率高!
  45. */
  46. private static void iteratorTreeMapByEntryset(TreeMap map) {
  47. if (map == null)
  48. return ;
  49. System.out.println("\niterator TreeMap By entryset");
  50. String key = null;
  51. Integer integ = null;
  52. Iterator iter = map.entrySet().iterator();
  53. while(iter.hasNext()) {
  54. Map.Entry entry = (Map.Entry)iter.next();
  55. key = (String)entry.getKey();
  56. integ = (Integer)entry.getValue();
  57. System.out.println(key+" -- "+integ.intValue());
  58. }
  59. }
  60. /*
  61. * 通过keyset来遍历TreeMap
  62. * 效率低!
  63. */
  64. private static void iteratorTreeMapByKeyset(TreeMap map) {
  65. if (map == null)
  66. return ;
  67. System.out.println("\niterator TreeMap By keyset");
  68. String key = null;
  69. Integer integ = null;
  70. Iterator iter = map.keySet().iterator();
  71. while (iter.hasNext()) {
  72. key = (String)iter.next();
  73. integ = (Integer)map.get(key);
  74. System.out.println(key+" -- "+integ.intValue());
  75. }
  76. }
  77. /*
  78. * 遍历TreeMap的values
  79. */
  80. private static void iteratorTreeMapJustValues(TreeMap map) {
  81. if (map == null)
  82. return ;
  83. Collection c = map.values();
  84. Iterator iter= c.iterator();
  85. while (iter.hasNext()) {
  86. System.out.println(iter.next());
  87. }
  88. }
  89. }

五.TreeMap示例

下面通过实例来学习如何使用TreeMap

  1. import java.util.*;
  2. /**
  3. * @desc TreeMap测试程序
  4. *
  5. * @author skywang
  6. */
  7. public class TreeMapTest {
  8. public static void main(String[] args) {
  9. // 测试常用的API
  10. testTreeMapOridinaryAPIs();
  11. // 测试TreeMap的导航函数
  12. //testNavigableMapAPIs();
  13. // 测试TreeMap的子Map函数
  14. //testSubMapAPIs();
  15. }
  16. /**
  17. * 测试常用的API
  18. */
  19. private static void testTreeMapOridinaryAPIs() {
  20. // 初始化随机种子
  21. Random r = new Random();
  22. // 新建TreeMap
  23. TreeMap tmap = new TreeMap();
  24. // 添加操作
  25. tmap.put("one", r.nextInt(10));
  26. tmap.put("two", r.nextInt(10));
  27. tmap.put("three", r.nextInt(10));
  28. System.out.printf("\n ---- testTreeMapOridinaryAPIs ----\n");
  29. // 打印出TreeMap
  30. System.out.printf("%s\n",tmap );
  31. // 通过Iterator遍历key-value
  32. Iterator iter = tmap.entrySet().iterator();
  33. while(iter.hasNext()) {
  34. Map.Entry entry = (Map.Entry)iter.next();
  35. System.out.printf("next : %s - %s\n", entry.getKey(), entry.getValue());
  36. }
  37. // TreeMap的键值对个数
  38. System.out.printf("size: %s\n", tmap.size());
  39. // containsKey(Object key) :是否包含键key
  40. System.out.printf("contains key two : %s\n",tmap.containsKey("two"));
  41. System.out.printf("contains key five : %s\n",tmap.containsKey("five"));
  42. // containsValue(Object value) :是否包含值value
  43. System.out.printf("contains value 0 : %s\n",tmap.containsValue(new Integer(0)));
  44. // remove(Object key) : 删除键key对应的键值对
  45. tmap.remove("three");
  46. System.out.printf("tmap:%s\n",tmap );
  47. // clear() : 清空TreeMap
  48. tmap.clear();
  49. // isEmpty() : TreeMap是否为空
  50. System.out.printf("%s\n", (tmap.isEmpty()?"tmap is empty":"tmap is not empty") );
  51. }
  52. /**
  53. * 测试TreeMap的子Map函数
  54. */
  55. public static void testSubMapAPIs() {
  56. // 新建TreeMap
  57. TreeMap tmap = new TreeMap();
  58. // 添加“键值对”
  59. tmap.put("a", 101);
  60. tmap.put("b", 102);
  61. tmap.put("c", 103);
  62. tmap.put("d", 104);
  63. tmap.put("e", 105);
  64. System.out.printf("\n ---- testSubMapAPIs ----\n");
  65. // 打印出TreeMap
  66. System.out.printf("tmap:\n\t%s\n", tmap);
  67. // 测试 headMap(K toKey)
  68. System.out.printf("tmap.headMap(\"c\"):\n\t%s\n", tmap.headMap("c"));
  69. // 测试 headMap(K toKey, boolean inclusive)
  70. System.out.printf("tmap.headMap(\"c\", true):\n\t%s\n", tmap.headMap("c", true));
  71. System.out.printf("tmap.headMap(\"c\", false):\n\t%s\n", tmap.headMap("c", false));
  72. // 测试 tailMap(K fromKey)
  73. System.out.printf("tmap.tailMap(\"c\"):\n\t%s\n", tmap.tailMap("c"));
  74. // 测试 tailMap(K fromKey, boolean inclusive)
  75. System.out.printf("tmap.tailMap(\"c\", true):\n\t%s\n", tmap.tailMap("c", true));
  76. System.out.printf("tmap.tailMap(\"c\", false):\n\t%s\n", tmap.tailMap("c", false));
  77. // 测试 subMap(K fromKey, K toKey)
  78. System.out.printf("tmap.subMap(\"a\", \"c\"):\n\t%s\n", tmap.subMap("a", "c"));
  79. // 测试
  80. System.out.printf("tmap.subMap(\"a\", true, \"c\", true):\n\t%s\n",
  81. tmap.subMap("a", true, "c", true));
  82. System.out.printf("tmap.subMap(\"a\", true, \"c\", false):\n\t%s\n",
  83. tmap.subMap("a", true, "c", false));
  84. System.out.printf("tmap.subMap(\"a\", false, \"c\", true):\n\t%s\n",
  85. tmap.subMap("a", false, "c", true));
  86. System.out.printf("tmap.subMap(\"a\", false, \"c\", false):\n\t%s\n",
  87. tmap.subMap("a", false, "c", false));
  88. // 测试 navigableKeySet()
  89. System.out.printf("tmap.navigableKeySet():\n\t%s\n", tmap.navigableKeySet());
  90. // 测试 descendingKeySet()
  91. System.out.printf("tmap.descendingKeySet():\n\t%s\n", tmap.descendingKeySet());
  92. }
  93. /**
  94. * 测试TreeMap的导航函数
  95. */
  96. public static void testNavigableMapAPIs() {
  97. // 新建TreeMap
  98. NavigableMap nav = new TreeMap();
  99. // 添加“键值对”
  100. nav.put("aaa", 111);
  101. nav.put("bbb", 222);
  102. nav.put("eee", 333);
  103. nav.put("ccc", 555);
  104. nav.put("ddd", 444);
  105. System.out.printf("\n ---- testNavigableMapAPIs ----\n");
  106. // 打印出TreeMap
  107. System.out.printf("Whole list:%s%n", nav);
  108. // 获取第一个key、第一个Entry
  109. System.out.printf("First key: %s\tFirst entry: %s%n",nav.firstKey(), nav.firstEntry());
  110. // 获取最后一个key、最后一个Entry
  111. System.out.printf("Last key: %s\tLast entry: %s%n",nav.lastKey(), nav.lastEntry());
  112. // 获取“小于/等于bbb”的最大键值对
  113. System.out.printf("Key floor before bbb: %s%n",nav.floorKey("bbb"));
  114. // 获取“小于bbb”的最大键值对
  115. System.out.printf("Key lower before bbb: %s%n", nav.lowerKey("bbb"));
  116. // 获取“大于/等于bbb”的最小键值对
  117. System.out.printf("Key ceiling after ccc: %s%n",nav.ceilingKey("ccc"));
  118. // 获取“大于bbb”的最小键值对
  119. System.out.printf("Key higher after ccc: %s%n\n",nav.higherKey("ccc"));
  120. }
  121. }

运行结果

{one=8, three=4, two=2}
next : one - 8
next : three - 4
next : two - 2
size: 3
contains key two : true
contains key five : false
contains value 0 : false
tmap:{one=8, two=2}
tmap is empty