关于TreeSet

 

java.lang.Object
   |_ java.util.AbstractCollection<E>
        |_ java.util.AbstractSet<E>
              |_ java.util.TreeSet<E>

TreeSet类声明如下:

public class TreeSet<E>
    extends AbstractSet<E>
    implements SortedSet<E>, Cloneable, java.io.Serializable

因为实现了SortedSet类,所以具有自然排序的功能。

TreeSet和HashSet相同的地方,就是集合里面不允许有重复的元素。自然排序情况下,一个TreeSet中只允许存放同一类型的多个元素,这里要求不是自定义的类。例如:

Set treeSet = new TreeSet();
   treeSet.add(new String("aaa"));
   treeSet.add(new String("aaa"));
   treeSet.add(new String("bbb"));
   treeSet.add(new String("ccc"));
   treeSet.add(new String("aaa"));
   System.out.println(treeSet);

结果输出为:

[aaa, bbb, ccc]

这时,treeSet.size()=3。而且,它是经过排序的输出。

如果有多个类的对象都加入到TreeSet集合中,就会发生异常。比如:

treeSet.add(new String("aaa"));
   treeSet.add(new Integer(100));
   System.out.println(treeSet);

发生异常:

Exception in thread "main" java.lang.ClassCastException: java.lang.String
at java.lang.Integer.compareTo(Integer.java:35)
at java.util.TreeMap.compare(TreeMap.java:1093)
at java.util.TreeMap.put(TreeMap.java:465)
at java.util.TreeSet.add(TreeSet.java:210)
at org.shirdrn.TreeSetTest.main(TreeSetTest.java:18)

而对于自定义的类,它的对象只能存放一个,而且实现类不需要实现Comparable接口。

但是,如果想要存放多个自定义的类的对象,不实现Comparable接口就会发生java.lang.ClassCastException异常。因此,想要能够进行客户化排序,必须实现比较器。

实现Comparable接口,就要实现compareTo()方法。而TreeSet又不存储相同的元素,这就要求自定义的类重写hashCode()和equals()方法:

class Person implements Comparable{
private String name;
private Integer age;
public Integer getAge() {
   return age;
}
public void setAge(Integer age) {
   this.age = age;
}
public String getName() {
   return name;
}
public void setName(String name) {
   this.name = name;
}

public boolean equals(Object o){
   if(this == o){
    return true;
   }
   if(! (o instanceof Person)){
    return false;
   }
   final Person other = (Person)o;
   if(this.name.equals(other.getName()) && this.age.equals(other.getAge())){
    return true;
   }
   else{
    return false;
   }
}

public int hashCode(){
   int result;
   result = (name == null?0:name.hashCode());
   result = 37*result + (age == null?0:age.hashCode());
   return result;
}

public int compareTo(Object o){
   Person other = (Person)o;
   if(this.name.compareTo(other.getName()) > 0){
    return 1;
   }
   if(this.name.compareTo(other.getName()) < 0){
    return -1;
   }
   if(this.getAge().intValue() > other.getAge().intValue()){
    return 1;
   }
   if(this.getAge().intValue() < other.getAge().intValue()){
    return -1;
   }
   return 0;
}

}

测试一下:

Set treeSet = new TreeSet();
   Person p1 = new Person();
   p1.setName("shirdrn");
   p1.setAge(new Integer(26));
   treeSet.add(p1);
   Person p2 = new Person();
   p2.setName("shirdrn");
   p2.setAge(new Integer(26));
   treeSet.add(p2);
   System.out.println(treeSet);

实例化了2个Person对象,实际上他们是同一个,因此只输出一个:

[org.shirdrn.Person@c29b5984]

如果将p2的name设置为p2.setName("Keller"),则输出两个:

[org.shirdrn.Person@5462263d, org.shirdrn.Person@c29b5984]

由于在Person类中实现类了compareTo()方法,输出结果是排序的,首先按照name排序,然后再按照age排序:

while(it.hasNext()){
    Person p = (Person)it.next();
    System.out.println("name = "+p.getName()+" || age = "+p.getAge());
   }

输出结果为:

name = Keller || age = 26
name = shirdrn || age = 26

name按照字母序排序。如果name相同,就按照age数字序排序。

TreeSet具有一些和HashSet类似的方法。

 

PS:如果不想去重写又想插入自定义的类

也可以用 public boolean addAll(Collection<? extends E> c) 方法!

你可能感兴趣的:(关于TreeSet)