A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 莫道荣 中级黑马   /  2013-3-15 23:35  /  1626 人查看  /  3 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 莫道荣 于 2013-3-16 21:23 编辑

import java.util.*;
class Person {
  private String name;
  Person(String name) {
    this.name = name;
  }
  public String getName() {
    return name;
  }
}
class Student extends Person {
  Student(String name) {
    super(name);
  }
}
class Worker extends Person {
  Worker(String name) {
    super(name);
  }
}
class Comp implements Comparator<Person> { //这里可以接收子类对象
  public int compare(Person p1, Person p2) { //这里可以接收子类对象
    return p1.getName().compareTo(p2.getName());
  }
}
public class GenericDemo06 {
  public static void main(String[] args) {
    TreeSet<Student> t1 = new TreeSet<Student>(new Comp());
    t1.add(new Student("stu01"));
    t1.add(new Student("stu05"));
    t1.add(new Student("stu03"));

    TreeSet<Worker> t2 = new TreeSet<Worker>(new Comp());
    t2.add(new Worker("work 01"));
    t2.add(new Worker("work 07"));
    t2.add(new Worker("work 05"));

    printColl(t1);
    printColl(t2);
  }
  public static void printColl(TreeSet<? extends Person> t) { //接受 Person 及其子类
    Iterator<? extends Person> it = t.iterator();
    while(it.hasNext())
      System.out.println(it.next().getName());
  }
}

TreeSet(Comparator<? super E> comparator)
构造一个新的空 TreeSet,它根据指定比较器进行排序。

以上两行是API文档里查的 TreeSet<E> 的一个构造方法及其描述,里面写的是 <? super E>
意思是说,只能接受比较 E 或者 E 的父类的比较器,代码中 TreeSet 里传的是 Person类型,那不就是说传入的比较器只能接受比较 Person 类或者 Person 的父类的对象吗?
为什么能比较 Person 的子类 Student 和 Worker 类型呀???
这点有点晕,求大神指教,解释一下

点评

如果问题已经解决,请将分类改为“已解决”,谢谢  发表于 2013-3-16 00:39

评分

参与人数 1技术分 +1 收起 理由
洪建超 + 1

查看全部评分

3 个回复

倒序浏览
你可能搞错通配符的概念了。Java中的泛型和通配符初看起来似乎可以这么理解,其实不然。

首先,需要了解的是,泛型尖括号中的父子类关系,并不意味着泛型本身也存在父子类关系。举个例子,A extends B,那么Collection<A>并不是Collection<B>的子类。但是Set<A>却是Collection<A>的子类。

这么说来是很好理解的,但是,Collection<A>和Collection<B>确实有关系啊,如何表现这种关系呢?于是Java就引入了通配符这一概念,并引入通配符可以定义这么一个东西:

Collection<? extends B>,对于这个东西,它不是任何泛型的子类,但一切尖括号内的类只要继承了B,或本身就是B,则一定是它的子类。如此看来,Collection<A>和Collection<B>都是Collection<? extends B>的子类

评分

参与人数 1技术分 +1 收起 理由
洪建超 + 1

查看全部评分

回复 使用道具 举报
 TreeSet<Student> t1 = new TreeSet<Student>(new Comp()); //这里用来比较学生只是一个多态的思想,这里的只会当做Person 来看的,
 TreeSet<Worker> t2 = new TreeSet<Worker>(new Comp());// 同上:这里的Worker 只会当作Person 来看的。

class Comp implements Comparator<Person> {
  public int compare(Person p1, Person p2) {  //这里的参数就是多态,这里不管是传的是学生,还是工人都只会当作Person来看的。是把他们当作 Person 来 比较的。
    return p1.getName().compareTo(p2.getName());
  }
}

回答:代码中 TreeSet 里传的是 Person类型,那不就是说传入的比较器只能接受比较 Person 类或者 Person 的父类的对象吗?  为什么能比较 Person 的子类 Student 和 Worker 类型呀???

注意:你理解的  这里只能接受比较 Person 类或者 Person 的父类的对象 是对的,但是你要记清多态思想,可以将子类对象 提升为父类用,只是要失去子类的特有行为和属性,
      所以TreeSet  里面我们进行比较 Student 和 Worker ,都是把他们当作Person来看的,
提示:你要留意  compare(Person p1, Person p2)  这个方法的参数类型。

评分

参与人数 1技术分 +1 收起 理由
洪建超 + 1

查看全部评分

回复 使用道具 举报
TreeSet(Comparator<? super E> comparator)
这个TreeSet的构造函数函数里面传的是一个比较器,这个比较器可以定义为泛型,  ? super E :可以接受E类型或者E的父类型,下限。
这个楼主说的没有错,因为比较器所接受的泛型可以为Person或者Person的父类,但是楼主往里面所传的已经是一个Person类了,那边比较器以及定义成功
那么既然父类都具备比较功能了,子类Student和Worker又继承了Person,它们也就同时具备了父类所拥有的比较器功能
这里主要涉及的是多态的问题,既然你比较器里面传的参数是Person,那边它的子类Student和Worker也同样可以传递给比较器
也就是是说你的class Comp implements Comparator<Person>  包含了
class Comp implements Comparator<Student> 及class Comp implements Comparator<Worker>
我想到这里你应该可以理解了吧
而接下来在集合的构造函数中传递一个已经定义好的比较器,那么楼主所传递的比较器是父子类都可以用的
在此再重新声明下TreeSet集合的两种排序方式:以便大家都复习下
TreeSet默认排序方式:让元素自身具备比较性,元素需要实现comparable接口,覆盖compareTo()方法。
TreeSet的第二种排序方式:当元素自身不具备比较性时,或者具备的比较性不是所需要的,这时就需要让集合自身具备比较性。
    在集合初始化时,就有了比较方式。定义了比较器,将比较器对象作为参数传递给TreeSet集合的构造函数


评分

参与人数 1技术分 +1 收起 理由
洪建超 + 1

查看全部评分

回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马