泛型之extends通配符
泛型的继承关系:Pair<Integer>
不是Pair<Number>
的子类。
extends通配符
1 import java.util.*; 2 public class Demo13{ 3 4 public static void main(String[] args) throws Exception{ 5 //实例化一个Pair<Integer>泛型类 6 Pair<Integer> p = new Pair<>(12,34); 7 int n = add(p); 8 System.out.println(n); 9 } 10 11 //此静态方法形参是传入一个Pair<Number>的泛型实例 12 static int add(Pair<Number> p){ 13 Number first = p.getFirst(); 14 Number last = p.getLast(); 15 //Number类型使用intValue()返回int类型 16 return first.intValue() + last.intValue(); 17 } 18 } 19 20 //定义一个泛型类 21 class Pair<T> { 22 private T first; 23 private T last; 24 public Pair(T first,T last){ 25 this.first = first; 26 this.last = last; 27 } 28 public T getFirst(){ 29 return first; 30 } 31 public T getLast(){ 32 return last; 33 } 34 }
原因很明显,因为Pair<Integer>
不是Pair<Number>
的子类,因此,add(Pair<Number>)
不接受参数类型Pair<Integer>
。
但是从add()
方法的代码可知,传入Pair<Integer>
是完全符合内部代码的类型规范,因为语句:
Number first = p.getFirst();
Number last = p.getLast();
实际类型是Integer
,引用类型是Number
,没有问题。问题在于方法参数类型定死了只能传入Pair<Number>
。
有没有办法使得方法参数接受Pair<Integer>
?办法是有的,这就是使用Pair<? extends Number>
使得方法接收所有泛型类型为Number
或Number
子类的Pair
类型。我们把代码局部改写如下:
//此静态方法形参是传入一个Pair<Number>的泛型实例 //static int add(Pair<Number> p){ //形参改为传入一个Pair<? extends Number>的泛型实例, //Pair<? extends Number>使得方法接收所有泛型类型为Number或Number子类的Pair类型 static int add(Pair<? extends Number> p){ Number first = p.getFirst(); Number last = p.getLast(); //Number类型使用intValue()返回int类型 return first.intValue() + last.intValue(); }
这样一来,给方法传入Pair<Integer>
类型时,它符合参数Pair<? extends Number>
类型。这种使用<? extends Number>
的泛型定义称之为上界通配符(Upper Bounds Wildcards),即把泛型类型T
的上界限定在Number
了。
除了可以传入Pair<Integer>
类型,我们还可以传入Pair<Double>
类型,Pair<BigDecimal>
类型等等,因为Double
和BigDecimal
都是Number
的子类。
如果我们考察对Pair<? extends Number>
类型调用getFirst()
方法,实际的方法签名变成了:
<? extends Number> getFirst();
即返回值是Number
或Number
的子类,因此,可以安全赋值给Number
类型的变量:
Number x = p.getFirst();
然后,我们不可预测实际类型就是Integer
,例如,下面的代码是无法通过编译的:
Integer x = p.getFirst();
这是因为实际的返回类型可能是Integer
,也可能是Double
或者其他类型,编译器只能确定类型一定是Number
的子类(包括Number
类型本身),但具体类型无法确定。
我们再来考察一下Pair<T>
的set
方法:
1 package day1_8; 2 3 public class Extends_Super { 4 public static void main(String[] args) { 5 Pair<Integer> p = new Pair<>(12,34); 6 System.out.println(add(p)); 7 } 8 static int add(Pair<? extends Number> p){ 9 Number first = p.getFirst(); 10 Number last = p.getLast(); 11 p.setFirst(new Integer(first.intValue()+100)); 12 p.setLast(new Integer(last.intValue() + 100)); 13 return p.getFirst().intValue() + p.getLast().intValue(); 14 } 15 } 16 17 class Pair<T> { 18 private T first; 19 private T last; 20 21 public Pair(T first, T last) { 22 this.first = first; 23 this.last = last; 24 } 25 26 public T getFirst() { 27 return first; 28 } 29 public T getLast() { 30 return last; 31 } 32 public void setFirst(T first) { 33 this.first = first; 34 } 35 public void setLast(T last) { 36 this.last = last; 37 } 38 }
不出意外,我们会得到一个编译错误:
编译错误发生在p.setFirst()
传入的参数是Integer
类型。有些童鞋会问了,既然p
的定义是Pair<? extends Number>
,那么setFirst(? extends Number)
为什么不能传入Integer
?
原因还在于擦拭法。如果我们传入的p
是Pair<Double>
,显然它满足参数定义Pair<? extends Number>
,然而,Pair<Double>
的setFirst()
显然无法接受Integer
类型。
这就是<? extends Number>
通配符的一个重要限制:方法参数签名setFirst(? extends Number)
无法传递任何Number
的子类型给setFirst(? extends Number)
。
这里唯一的例外是可以给方法参数传入null
:
p.setFirst(null); // ok, 但是后面会抛出NullPointerException p.getFirst().intValue(); // NullPointerException
extends通配符的作用
如果我们考察Java标准库的java.util.List<T>
接口,它实现的是一个类似“可变数组”的列表,主要功能包括:
public interface List<T> { int size(); // 获取个数 T get(int index); // 根据索引获取指定元素 void add(T t); // 添加一个新元素 void remove(T t); // 删除一个已有元素 }
现在,让我们定义一个方法来处理列表的每个元素:
1 package day1_8; 2 3 import java.util.ArrayList; 4 import java.util.List; 5 6 public class Extends_Super { 7 public static void main(String[] args) { 8 ArrayList<Integer> list = new ArrayList<>(); 9 System.out.println(sumOfList(list)); 10 } 11 //注意形参处是List<Integer> 12 static int sumOfList(List<Integer> list){ 13 int sum = 0; 14 for (int i = 0; i < 101; i++) { 15 list.add(i); //可以对list写 16 sum += list.get(i); //可以对list读 17 } 18 return sum; 19 } 20 }
运行完全OK
接下来我们改动一处再看看
1 package day1_8; 2 import java.util.ArrayList; 3 import java.util.List; 4 5 public class Extends_Super { 6 public static void main(String[] args) { 7 ArrayList<Integer> list = new ArrayList<>(); 8 System.out.println(sumOfList(list)); 9 } 10 11 //注意形参改为<? extends Integer>通配符 12 static int sumOfList(List<? extends Integer> list){ 13 int sum = 0; 14 for (int i = 0; i < 101; i++) { 15 //下面两行是一样的,因为Integer和int是自动拆装和包装的 16 //list.add(i); //不能对list写 17 list.add(new Integer(i)); //不能对list写 18 sum += list.get(i); //可以对list读 19 } 20 return sum; 21 } 22 }
为什么我们定义的方法参数类型是List<? extends Integer>
而不是List<Integer>
?从方法内部代码看,传入List<? extends Integer>
或者List<Integer>
是完全一样的,但是,注意到List<? extends Integer>
的限制:
- 允许调用
get()
方法获取Integer
的引用; - 不允许调用
set(? extends Integer)
方法并传入任何Integer
的引用(null
除外)。
因此,方法参数类型List<? extends Integer>
表明了该方法内部只会读取List
的元素,不会修改List
的元素(因为无法调用add(? extends Integer)
、remove(? extends Integer)
这些方法。换句话说,这是一个对参数List<? extends Integer>
进行只读的方法(恶意调用set(null)
除外)。
使用extends限定T类型
在定义泛型类型Pair<T>
的时候,也可以使用extends
通配符来限定T
的类型:
public class Pair<T extends Number> { ... }
现在,我们只能定义:
Pair<Number> p1 = null; Pair<Integer> p2 = new Pair<>(1, 2); Pair<Double> p3 = null;
因为Number
、Integer
和Double
都符合<T extends Number>
。
非Number
类型将无法通过编译:
Pair<String> p1 = null; // compile error! Pair<Object> p2 = null; // compile error!
因为String
、Object
都不符合<T extends Number>
,因为它们不是Number
类型或Number
的子类。
小结
使用类似<? extends Number>
通配符作为方法参数时表示:
-
方法内部可以调用获取
Number
引用的方法,例如:Number n = obj.getFirst();
; -
方法内部无法调用传入
Number
引用的方法(null
除外),例如:obj.setFirst(Number n);
。
即一句话总结:使用extends
通配符表示可以读,不能写。
使用类似<T extends Number>
定义泛型类时表示:
- 泛型类型限定为
Number
以及Number
的子类。
这里唯一的例外是可以给方法参数传入null
: