问题
(1)TreeSet真的是使用TreeMap来存储元素的吗?
(2)TreeSet是有序的吗?
(3)TreeSet和LinkedHashSet有何不同?
简介
TreeSet底层是采用TreeMap实现的一种Set,所以它是有序的,同样也是非线程安全的。
UML继承关系
源码分析
经过前面我们学习HashSet和LinkedHashSet,基本上已经掌握了Set实现的套路了。
所以,也不废话了,直接上源码:
1 | package java.util; |
源码比较简单,基本都是调用map相应的方法。
总结
(1)TreeSet底层使用NavigableMap存储元素;
(2)TreeSet是有序的;
(3)TreeSet是非线程安全的;
(4)TreeSet实现了NavigableSet接口,而NavigableSet继承自SortedSet接口;
(5)TreeSet实现了SortedSet接口;(彤哥年轻的时候面试被问过TreeSet和SortedSet的区别^^)
彩蛋
(1)通过之前的学习,我们知道TreeSet和LinkedHashSet都是有序的,那它们有何不同?
LinkedHashSet并没有实现SortedSet接口,它的有序性主要依赖于LinkedHashMap的有序性,所以它的有序性是指按照插入顺序保证的有序性;
而TreeSet实现了SortedSet接口,它的有序性主要依赖于NavigableMap的有序性,而NavigableMap又继承自SortedMap,这个接口的有序性是指按照key的自然排序保证的有序性,而key的自然排序又有两种实现方式,一种是key实现Comparable接口,一种是构造方法传入Comparator比较器。
(2)TreeSet里面真的是使用TreeMap来存储元素的吗?
通过源码分析我们知道TreeSet里面实际上是使用的NavigableMap来存储元素,虽然大部分时候这个map确实是TreeMap,但不是所有时候都是TreeMap。
因为有一个构造方法是TreeSet(NavigableMap<E,Object> m)
,而且这是一个非public方法,通过调用关系我们可以发现这个构造方法都是在自己类中使用的,比如下面这个:
1 | public NavigableSet<E> tailSet(E fromElement, boolean inclusive) { |
而这个m我们姑且认为它是TreeMap,也就是调用TreeMap的tailMap()方法:
1 | public NavigableMap<K,V> tailMap(K fromKey, boolean inclusive) { |
可以看到,返回的是AscendingSubMap对象,这个类的继承链是怎么样的呢?
可以看到,这个类并没有继承TreeMap,不过通过源码分析也可以看出来这个类是组合了TreeMap,也算和TreeMap有点关系,只是不是继承关系。
所以,TreeSet的底层不完全是使用TreeMap来实现的,更准确地说,应该是NavigableMap。