北京北大青鳥學校學術部講師將繼續講解關于Java泛型的技術知識,關于Java泛型的定義、相關例子等請參考之前的幾篇文章,在這里就不做陳述了。今天這篇文章,將主講介紹一下Java類庫中的泛型(北大青鳥課程)
集合類
北京北大青鳥學校專家介紹:到目前為止,Java 類庫中泛型支持存在最多的地方就是集合框架。就像容器類是 C++ 語言中模板的主要動機一樣(參閱 附錄 A:與 C++ 模板的比較)(盡管它們隨后用于很多別的用途),改善集合類的類型安全是 Java 語言中泛型的主要動機。集合類也充當如何使用泛型的模型,因為它們演示了泛型的幾乎所有的標準技巧和方言。(北大青鳥課程)
所有的標準集合接口都是泛型化的 —— Collection
集合類也使用泛型的許多“技巧”和方言,比如上限通配符和下限通配符。例如,在接口 Collection
interface Collection
boolean addAll(Collection extends V>);
}
該定義組合了通配符類型參數和有限制類型參數,允許您將 Collection
北京北大青鳥學校專家提醒:如果類庫將 addAll() 定義為接受 Collection
應該可以將 addAll() 的類型參數定義為 Collection
作為泛型化一個類(如果不小心的話)如何會更改其語義的一個更加微妙的例子,注意 Collection.removeAll() 的參數的類型是 Collection>,而不是 Collection extends V>。這是因為傳遞混合類型的集合給 removeAll() 是可接受的,并且更加限制地定義 removeAll 將會更改方法的語義和有用性。(北大青鳥課程)
其他容器類
北京北大青鳥學校專家介紹:除了集合類之外,Java 類庫中還有幾個其他的類也充當值的容器。這些類包括 WeakReference、SoftReference 和 ThreadLocal。它們都已經在其包含的值的類型上泛型化了,所以 WeakReference
泛型不止用于容器
泛型最常見最直觀的使用是容器類,比如集合類或引用類(比如 WeakReference
像 Comparable
Comparable
Comparable 接口已經泛型化了,所以實現 Comparable 的對象聲明它可以與什么類型進行比較。北京北大青鳥學校專家總結:(通常,這是對象本身的類型,但是有時也可能是父類。)
public interface Comparable
public boolean compareTo(T other);
}
所以 Comparable 接口包含一個類型參數 T,該參數是一個實現 Comparable 的類可以與之比較的對象的類型。這意味著如果定義一個實現 Comparable 的類,比如 String,就必須不僅聲明類支持比較,還要聲明它可與什么比較(通常是與它本身比較):
public class String implements Comparable
現在來考慮一個二元 max() 方法的實現。您想要接受兩個相同類型的參數,二者都是 Comparable,并且相互之間是 Comparable。幸運的是,如果使用泛型方法和有限制類型參數的話,這相當直觀:
public static
if (t1.compareTo(t2) > 0)
return t1;
else
return t2;
}
北京北大青鳥學校專家介紹:在本例中,您定義了一個泛型方法,在類型 T 上泛型化,您約束該類型擴展(實現) Comparable
更好的是,編譯器將使用類型推理來確定當調用 max() 時 T 的值表示什么意思。所以根本不用指定 T,下面的調用就能工作:
String s = max("moo", "bark");
編譯器將計算出 T 的預定值是 String,因此它將進行編譯和類型檢查。但是如果您試圖用不實現 Comparable
Class
類 Class 已經泛型化了,但是很多人一開始都感覺其泛型化的方式很混亂。Class
在以前的 JDK 中,Class.newInstance() 方法的定義返回 Object,您很可能要將該返回類型強制轉換為另一種類型:
class Class {
Object newInstance();
}
但是使用泛型,您定義 Class.newInstance() 方法具有一個更加特定的返回類型:
class Class
T newInstance();
}(北大青鳥課程)
如何創建一個 Class
讓 Foo.class 是 Class
考慮一個方法,它從數據庫檢索一組對象,并返回 JavaBeans 對象的一個集合。您通過反射來實例化和初始化創建的對象,但是這并不意味著類型安全必須完全被拋至腦后。考慮下面這個方法:
public static
// Use Selector to select rows
List
for (/* iterate over results */) {
T row = c.newInstance();
// use reflection to set fields from result
list.add(row);
}
return list;
}
(北大青鳥課程)
可以像下面這樣簡單地調用該方法:
List
編譯器將會根據 FooRecord.class 是 Class
用 Class
Collection 接口包含一個方法,用于將集合的內容復制到一個調用者指定類型的數組中:
public Object[] toArray(Object[] prototypeArray) { ... }
toArray(Object[]) 的語義是,如果傳遞的數組足夠大,就會使用它來保存結果,否則,就會使用反射分配一個相同類型的新數組。一般來說,單獨傳遞一個數組作為參數來提供想要的返回類型是一個小技巧,但是在引入泛型之前,這是與方法交流類型信息最方便的方式。
有了泛型,就可以用一種更加直觀的方式來做這件事。不像上面這樣定義 toArray(),泛型 toArray() 可能看起來像下面這樣:
public
調用這樣一個 toArray() 方法很簡單:
FooBar[] fba = something.toArray(FooBar.class);
Collection 接口還沒有改變為使用該技術,因為這會破壞許多現有的集合實現。但是如果使用泛型從新構建 Collection,則當然會使用該方言來指定它想要返回值是哪種類型。
Enum
JDK 5.0 中 Java 語言另一個增加的特性是枚舉。當您使用 enum 關鍵字聲明一個枚舉時,編譯器就會在內部為您生成一個類,用于擴展 Enum 并為枚舉的每個值聲明靜態實例。所以如果您說:
public enum Suit {HEART, DIAMOND, CLUB, SPADE};
編譯器就會在內部生成一個叫做 Suit 的類,該類擴展 java.lang.Enum
與 Class 一樣,Enum 也是一個泛型類。但是與 Class 不同,它的簽名稍微更復雜一些:
class Enum
這究竟是什么意思?這難道不會導致無限遞歸?
我們逐步來分析。類型參數 E 用于 Enum 的各種方法中,比如 compareTo() 或 getDeclaringClass()。為了這些方法的類型安全,Enum 類必須在枚舉的類上泛型化。
所以 extends Enum
北京北大青鳥學校專家介紹,該部分又具有兩個部分。第一部分指出,作為 Enum 的類型參數的類本身必須是 Enum 的子類型,所以您不能聲明一個類 X 擴展 Enum
總之,Enum 是一個參數化的類型,只可以為它的子類型實例化,并且這些子類型然后將根據子類型來繼承方法。幸運的是,在 Enum 情況下,編譯器為您做這些工作,一切都很好。(北京北大青鳥學校,未完待續)