有時你可能但願限制可用做參數化類型中的類型參數的類型,例如,對數字進行操做的方法可能只想接受Number
或其子類的實例,這是有界類型參數的用途。java
要聲明有界類型參數,請列出類型參數的名稱,而後是extends
關鍵字,後跟其上限,在此示例中爲Number
,請注意,在此上下文中,extends
在通常意義上用於表示「extends」(如在類中)或「implements」(如在接口中)。算法
public class Box<T> { private T t; public void set(T t) { this.t = t; } public T get() { return t; } public <U extends Number> void inspect(U u){ System.out.println("T: " + t.getClass().getName()); System.out.println("U: " + u.getClass().getName()); } public static void main(String[] args) { Box<Integer> integerBox = new Box<Integer>(); integerBox.set(new Integer(10)); integerBox.inspect("some text"); // error: this is still String! } }
經過修改咱們的泛型方法來包含這個有界類型參數,編譯如今將失敗,由於咱們的inspect
調用仍然包含一個String
:segmentfault
Box.java:21: <U>inspect(U) in Box<java.lang.Integer> cannot be applied to (java.lang.String) integerBox.inspect("10"); ^ 1 error
除了限制可用於實例化泛型類型的類型以外,有界類型參數還容許你調用邊界中定義的方法:數組
public class NaturalNumber<T extends Integer> { private T n; public NaturalNumber(T n) { this.n = n; } public boolean isEven() { return n.intValue() % 2 == 0; } // ... }
isEven
方法經過n
調用Integer
類中定義的intValue
方法。app
前面的示例說明了使用帶有單個邊界的類型參數,可是類型參數能夠有多個邊界:this
<T extends B1 & B2 & B3>
具備多個邊界的類型變量是邊界中列出的全部類型的子類型,若是其中一個邊界是類,則必須首先指定它,例如:code
Class A { /* ... */ } interface B { /* ... */ } interface C { /* ... */ } class D <T extends A & B & C> { /* ... */ }
若是未首先指定邊界A
,則會出現編譯時錯誤:對象
class D <T extends B & A & C> { /* ... */ } // compile-time error
有界類型參數是通用算法實現的關鍵,請考慮如下方法,該方法計算數組T[]
中大於指定元素elem
的元素數。繼承
public static <T> int countGreaterThan(T[] anArray, T elem) { int count = 0; for (T e : anArray) if (e > elem) // compiler error ++count; return count; }
該方法的實現很簡單,但它不能編譯,由於大於運算符(>
)僅適用於基本類型,如short
、int
、double
、long
、float
、byte
和char
,你不能使用>
運算符來比較對象,要解決此問題,請使用由Comparable<T>
接口限定的類型參數:接口
public interface Comparable<T> { public int compareTo(T o); }
生成的代碼將是:
public static <T extends Comparable<T>> int countGreaterThan(T[] anArray, T elem) { int count = 0; for (T e : anArray) if (e.compareTo(elem) > 0) ++count; return count; }