下面一段話來自一位知乎用戶的回答,特摘抄如下:
假設(shè)現(xiàn)在有這么一個(gè)類的繼承樹垃它,Plant -> Fruit -> Apple -> RedApple。
List<? extends Fruit> appList2的意思,是一個(gè)列表国拇,這個(gè)列表里面的元素是Fruit的某個(gè)子類T洛史,那么在從appList2中取出一個(gè)元素時(shí),編譯器會(huì)自動(dòng)把這個(gè)元素轉(zhuǎn)型為T酱吝。那么現(xiàn)在假設(shè)T是RedApple也殖,很顯然你往這樣一個(gè)appList2中add一個(gè)Apple類型的元素,取出后轉(zhuǎn)型為RedApple必然會(huì)失斘袢取忆嗜;同樣的情況如果T是Apple,你add一個(gè)Fruit類型的元素崎岂,取出后轉(zhuǎn)型為Apple捆毫,也會(huì)拋出異常。也就是說冲甘,編譯器會(huì)把列表中取出的元素轉(zhuǎn)型為某種類型绩卤,但編譯器不確定這種轉(zhuǎn)型是不是會(huì)成功,即在不保證運(yùn)行時(shí)能順利進(jìn)行江醇,因此就不允許你add任何類型的元素濒憋。
再來看看List<? super Fruit> appList,這個(gè)列表的元素都是Fruit的父類T陶夜,也就是說當(dāng)你從這個(gè)列表中g(shù)et一個(gè)元素時(shí)凛驮,編譯器會(huì)自動(dòng)加一句轉(zhuǎn)型為T的代碼。好条辟,現(xiàn)在你往appList中add一個(gè)Apple類型的元素黔夭,取出時(shí)轉(zhuǎn)型為T,由于T是Apple的父類捂贿,向上轉(zhuǎn)型沒有問題纠修;加一個(gè)RedApple類型的元素,也一樣不會(huì)有問題厂僧。也就是說扣草,只要保證你往appList中添加的元素是Fruit的子類,編譯器就可以保證在轉(zhuǎn)型為T時(shí)不會(huì)拋出異常颜屠。因此第二種寫法可以過編譯辰妙。
還是用前面文章(http://www.reibang.com/p/5633ab144569)學(xué)生排序的例子。
學(xué)生類:
package Compara;
public class Student {
//定義學(xué)生類的名字name和分?jǐn)?shù)mark屬性
public String name;
public int mark;
//定義構(gòu)造方法
public Student(String name, int mark) {
super();
this.name = name;
this.mark = mark;
}
}
自定義的比較器StudentComparator:
package Compara;
import java.util.Comparator;
public class StudentComparator implements Comparator<Student> {
@Override
public int compare(Student o1, Student o2) {
if (o1.name.compareTo(o2.name) > 0) {
return 1;
} else if (o1.name.compareTo(o2.name) == 0) {
if (o1.mark - o2.mark > 0) {
return 1;
} else if (o1.mark - o2.mark == 0) {
return 0;
} else {
return -1;
}
} else {
return -1;
}
}
}
再自定義一個(gè)女學(xué)生類甫窟,繼承學(xué)生類:
package Compara;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class GirlStudent extends Student{
public String name;
public int mark;
public String sex="women";
public GirlStudent(String name, int mark) {
super(name, mark);
}
public static void main(String[] args) {
//new出來四個(gè)學(xué)生對(duì)象
GirlStudent girlstu1=new GirlStudent("Limei",568);
GirlStudent girlstu2=new GirlStudent("Chenxiaojuan",512);
GirlStudent girlstu3=new GirlStudent("Limei",620);
GirlStudent girlstu4=new GirlStudent("Zhouli",522);
//new一個(gè)list密浑,將四個(gè)學(xué)生扔進(jìn)去。
List<Student> girlstudentlist=new ArrayList<Student>();
girlstudentlist.add(girlstu1);
girlstudentlist.add(girlstu2);
girlstudentlist.add(girlstu3);
girlstudentlist.add(girlstu4);
//條用Collections下的sort方法進(jìn)行排序粗井。
Collections.sort(girlstudentlist,new StudentComparator());
//打印出排序后的情況
for(Student stu:girlstudentlist){
System.out.println(stu.name+"=>"+stu.mark);
}
}
}
結(jié)果:
Chenxiaojuan=>512
Limei=>568
Limei=>620
Zhouli=>522
我們看源碼Collecctions中sort方法:
public static <T> void sort(List<T> list, Comparator<? super T> c) {
list.sort(c);
}
Comparator<? super T> 表示是T的父類尔破?的比較器街图。這里的?在例子中指的是Student類,它是GirlStudent類的父類懒构。