首页 > 编程知识 正文

java怎么return字符串,java中的多态是怎么实现的

时间:2023-05-05 17:45:24 阅读:111378 作者:2037

本文介绍了TreeSet如何在Java中实现。 (详细调查),有一定的参考价值。 需要的朋友请参考。 我希望对你有帮助。

HashSet是基于HashMap实现的,那么TreeSet如何实现呢? 没错! 正如大家想的那样,它是基于TreeMap实现的。 所以,TreeSet的源代码也很简单,主要了解TreeMap。

TreeSet的继承关系

按照惯例,我们先看看TreeSet类的继承关系。

publicclasstreesetextendsabstractset

implements NavigableSet,Cloneable,java.io.Serializable

意外继承抽象类AbstracSet,容易扩展

实现了与NavigableMap界面类似的NavigableSet界面,提供了各种导航方法;

可以实现Cloneable接口并进行克隆;

实现了可序列化接口,可以序列化;

这里主要看NavigableSet接口类:

publicinterfacenavigablesetextendssortedset

熟悉的味道,继承SortedSet接口。 SortedSet提供了返回比较器的方法。

Comparator super E comparator (;

和SortedMap一样,支持自然排序和自定义排序。 对于自然排序,添加到Set中的元素必须实现Comparable接口,而自定义排序则必须实现Comparator比较器。

源代码分析

关键点

关键当然是TreeSet如何保证元素不重叠和元素有序,如上所述是基于TreeMap实现的,让我们来看看吧。

私有传输navigable map m; //保持秩序

//dummyvaluetoassociatewithanobjectinthebackingmap

privatestaticfinalobjectpresent=new object (; //固定值

通过查看TreeSet源代码,我们发现只有这两个属性(也有uid,但在这里不计算) 显然,m用于存储元素,但m声明的是NavigableMap而不是TreeMap。 据推测,TreeMap应该是用构建方法实例化的,这里可以通过使用NavigableMap提高TreeSet的灵活性。 PRESENT与HashSet的PRESENT一样,作为固定的Value值被占位符。

让我们看看add和remove的方法:

publicbooleanadd{

returnm.put(e,PRESENT )==null;

}

公共空间内存(objecto ) {

returnm.remove(o )==PRESENT;

}

与HashSet的实现相同,还具有利用Map保存的Key-Value键值对的Key不重复的特征。

构造函数

还是用构造函数初始化TreeSet的TreeMap。

公共树集

this (新流) ); //默认自然排序的TreeMap

}

公共流集(comparatorsuperecomparator ) {

新流计算机(this ); //定制比较器流映射

}

公共流集(collectionextendsec ) {

this (; //还是默认使用

addall(c ); 将//元素逐个添加到TreeMap

}

公共流集(sorted sets ) {

this(s.comparator ) ); //使用传入的SortedSet的比较器

addall(s ); //一个一个地添加元素

}

缺省情况下,会实例化自然排序的TreeMap。 当然,可以定制比较器。

在这里追踪addAll方法。

publicbooleanaddall (collectionextendsec ) (

//use linear-timeversionifapplicable

if(m.size(==0c.size ) ) 0

c instanceof SortedSet

m instanceof TreeMap )

{

SortedSet extends E> set = (SortedSet extends E>) c;

TreeMap map = (TreeMap) m; // 强转成TreeMap

Comparator> cc = set.comparator();

Comparator super E> mc = map.comparator();

if (cc==mc || (cc != null && cc.equals(mc))) { // 要保证set和map的比较器一样

map.addAllForTreeSet(set, PRESENT); // TreeMap专门为TreeSet准备的方法

return true;

}

}

return super.addAll(c);

}

调用了TreeMap的addAllForTreeSet方法:

void addAllForTreeSet(SortedSet extends K> set, V defaultVal) {

try {

buildFromSorted(set.size(), set.iterator(), null, defaultVal);

} catch (java.io.IOException | ClassNotFoundException cannotHappen) {

}

}

看到buildFromSorted,应该很熟悉,在TreeMap的文章中分析过。该方法将传入的集合元素构造成了一棵最底层的结点为红色,而其他结点都是黑色的红黑树。

导航方法

既然实现了NavigableSet,那各种导航方法自然少不了。它们的实现也很简单,直接调用m对应的导航方法即可。例如:

public E first() {

return m.firstKey(); // 返回第一个元素

}

public E lower(E e) {

return m.lowerKey(e); // 返回小于e的第一个元素

}

public NavigableSet headSet(E toElement, boolean inclusive) {

return new TreeSet<>(m.headMap(toElement, inclusive)); // 取前几个元素构成子集

}

public E pollFirst() { // 弹出第一个元素

Map.Entry e = m.pollFirstEntry();

return (e == null) ? null : e.getKey();

}

public NavigableSet descendingSet() { // 倒排Set

return new TreeSet<>(m.descendingMap());

}

......

这里需要注意的是返回子集合的方法,例如:headSet。返回的子集合是可以添加和删除元素的,但是有边界限制,举个栗子。

// 前面构造了一个存储Int的Set

// 3、5、7、9

SortedSet subSet = intSet.headSet(8); // 最大值7,超过7越界

for (Integer sub : subSet) {

System.out.println(sub);

}

subSet.add(2);

// subSet.add(8); // 越界了

subSet.remove(3);

for (Integer sub : subSet) {

System.out.println(sub);

}

TreeSet也是支持逆序输出的,因为有descendingIterator的实现:

public Iterator descendingIterator() {

return m.descendingKeySet().iterator();

}

总结

TreeSet是基于TreeMap实现的,支持自然排序和自定义排序,可以进行逆序输出;

TreeSet不允许null值;

TreeSet不是线程安全的,多线程环境下可以使用SortedSet s = Collections.synchronizedSortedSet(new TreeSet(...));

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。