有时你可能但愿限制可用做参数化类型中的类型参数的类型,例如,对数字进行操做的方法可能只想接受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; }