要了解協變與逆變,首先要引入:
根據Liskov替換原則,如果C是P的子類,則P可以代替C,即P p = new C();
什麼是協變
C繼承於P,記做為C dc667ae6dfa312e15951fa8a71fc2b8c和
Listc0f559cc8d56b43654fcbe4aa9df7b4a不存在任何繼承關係
如果F是協變的,當C 8af599b3fba6d3d367fcf89625567c3c可以看作為
ArrayListc0f559cc8d56b43654fcbe4aa9df7b4a的父類別
??extend Number
陣列預設是協變的可以看作為一個型別範圍,表示Number的某一個子類別
Number[] numbers = new Integer[3];什麼是逆變如果F是逆變的,當C 791abd30474f2f144e6de456a95a431d= F(P)Java 提供了一個super來將不變轉為協變,例如:
List<? super Number> list = new ArrayList<Object>(1); //corrent此時的Listda50108ad159903fabe211f1543600e8
#extends 和super可以看身為
ArrayLista87fdacec66f0909fc0757c19f2d2b1d的父類別
首先,我們來看看Collection.add的實作:
public interface List<E> extends Collection<E> { boolean add(E e); }下面程式碼將會報錯? ? extends Number
與
Integer類型不符
List<? extends Number> list = new ArrayList<Integer>(); // correct list.add(Integer.valueOf(1)); //error首先在呼叫add方法時,泛型E
第二行報錯,也就是說自動變成了
a2b037db85f4e1df0e812b9647ac55a8? extends Number
不是
Integer的父類別。這裡要將
Lista2b037db85f4e1df0e812b9647ac55a8是
ArrayListc0f559cc8d56b43654fcbe4aa9df7b4a的父類別區分開。
? extends Number
可以看作為一個類型範圍中某一個類型,表示Number的某一個子類,但又沒明確是哪個子類,可能是Float,可能是Short ,也可能是Integer的子類別(Integer被final修飾,不可能有子類,這裡只是一種假設情況),它只確定了它的上界為Number,並沒有確定下界(有可能存在
?extends Number<
Integer),因此
#? extends Number不是
Integer的父類別
# 將上面程式碼稍做修改就正確了:
List<? super Number> list = new ArrayList<Object>(); // correct list.add(Integer.valueOf(1)); //correct首先因為逆變,Listda50108ad159903fabe211f1543600e8
第二行:是
ArrayLista87fdacec66f0909fc0757c19f2d2b1d的父類,第一行正確。
? super Number
#使用extends還是super呢是
Integer的父類,原因是:
? super Number表示Number的某一個父類,可能是
Serializable也可能是
Object但不管是哪個,Number的父類別一定是Integer的父類,因此第二行也正確
java.util.Collections的copy方法(JDK1.7)給了我們答案:
public static <T> void copy(List<? super T> dest, List<? extends T> src) { int srcSize = src.size(); if (srcSize > dest.size()) throw new IndexOutOfBoundsException("Source does not fit in dest"); if (srcSize < COPY_THRESHOLD || (src instanceof RandomAccess && dest instanceof RandomAccess)) { for (int i=0; i<srcSize; i++) dest.set(i, src.get(i)); } else { ListIterator<? super T> di=dest.listIterator(); ListIterator<? extends T> si=src.listIterator(); for (int i=0; i<srcSize; i++) { di.next(); di.set(si.next()); } } }
- 要從泛型類別取資料時,用extends;
- 要寫資料到泛型類別時,用super;
- 既要取又要寫,就不用通配符(即extends與super都不用)
private static <E> E getFirst(List<? extends E> list){ return list.get(0); } private static <E> void setFirst(List<? super E> list, E firstElement){ list.add(firstElement); } public static void main(String[] args) { List<Integer> list = new ArrayList<Integer>(); setFirst(list, 1); Number number = getFirst(list); }
以上是Java泛型之協變、逆變、extends與super選擇方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!