北京北大青鳥學(xué)校學(xué)術(shù)部講解:與非泛型代碼相互操作
北京北大青鳥學(xué)校學(xué)術(shù)部專家表示:數(shù)百萬行現(xiàn)有代碼使用已經(jīng)泛型化的Java類庫中的類,比如集合框架、Class 和 ThreadLocal。JDK 5.0 中的改進不要破壞所有這些代碼是很重要的,所以編譯器允許在不指定其類型參數(shù)的情況下使用泛型類。(北大青鳥課程)
當然,北京北大青鳥學(xué)校老師表示,以“舊方式”做事沒有新方式安全,因為忽略了編譯器準備提供的類型安全。如果試圖將 List<String> 傳遞給一個接受 List 的方法,它將能夠工作,但是編譯器將會發(fā)出一個可能喪失類型安全的警告,即所謂的“unchecked conversion(不檢查轉(zhuǎn)換)”警告。(北大青鳥課程)
沒有類型參數(shù)的泛型,比如聲明為 List 類型而不是 List<Something> 類型的變量,叫做原始類型。原始類型與參數(shù)化類型的任何實例化是賦值兼容的,但是這樣的賦值會生成 unchecked-conversion 警告。(北大青鳥課程)
為了消除一些 unchecked-conversion 警告,假設(shè)您不準備泛型化所有的代碼,您可以使用通配符類型參數(shù)。使用 List<?> 而不使用 List。List 是原始類型;List<?> 是具有未知類型參數(shù)的泛型。編譯器將以不同的方式對待它們,并很可能發(fā)出更少的警告。
北京北大青鳥學(xué)校老師介紹,無論在哪種情況下,編譯器在生成字節(jié)碼時都會生成強制類型轉(zhuǎn)換,所以生成的字節(jié)碼在每種情況下都不會比沒有泛型時更不安全。如果您設(shè)法通過使用原始類型或類文件來破壞類型安全,就會得到與不使用泛型時得到的相同的 ClassCastException 或 ArrayStoreException。(北大青鳥課程)
已檢查集合
作為從原始集合類型遷移到泛型集合類型的幫助,集合框架添加了一些新的集合包裝器,以便為一些類型安全bug提供早期警告。就像 Collections.unmodifiableSet() 工廠方法用一個不允許任何修改的 Set 包裝一個現(xiàn)有 Set 一樣,Collections.checkedSet()(以及 checkedList() 和 checkedMap())工廠方法創(chuàng)建一個包裝器(或者視圖)類,以防止您將錯誤類型的變量放在集合中。(北大青鳥課程)
checkedXxx() 方法都接受一個類常量作為參數(shù),所以它們可以(在運行時)檢查這些修改是允許的。典型的實現(xiàn)可能像下面這樣:
public class Collections {
public static <E> Collection<E> checkedCollection(Collection<E> c, Class<E> type ) {
return new CheckedCollection<E>(c, type);
}
private static class CheckedCollection<E> implements Collection<E> {
private final Collection<E> c;
private final Class<E> type;
CheckedCollection(Collection<E> c, Class<E> type) {
this.c = c;
this.type = type;
}
public boolean add(E o) {
if (!type.isInstance(o))
throw new ClassCastException();
else
return c.add(o);
}
}
} (北大青鳥課程)
北京北大青鳥學(xué)校提供