• 泛型之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>使得方法接收所有泛型类型为NumberNumber子类的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>类型等等,因为DoubleBigDecimal都是Number的子类。

    如果我们考察对Pair<? extends Number>类型调用getFirst()方法,实际的方法签名变成了:

    <? extends Number> getFirst();

    即返回值是NumberNumber的子类,因此,可以安全赋值给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

    原因还在于擦拭法。如果我们传入的pPair<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;

    因为NumberIntegerDouble都符合<T extends Number>

    Number类型将无法通过编译:

    Pair<String> p1 = null; // compile error!
    Pair<Object> p2 = null; // compile error!

    因为StringObject都不符合<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

  • 相关阅读:
    念大学有用么摘抄
    天行健,君子以自强不息;地势坤,君子以厚德载物。
    加快播放视频的播放速度
    微信语音通话中无法播放聊天对话框中的视频
    劝学
    在这个世界上就是物竞天择,适者生存。弱肉强食,优胜劣汰?
    英语名言名句集锦
    贵州理科状元邹基伟:不放过上课的每一秒
    带宽的理解
    第二章 Python运行程序
  • 原文地址:https://www.cnblogs.com/zui-ai-java/p/14252350.html
Copyright © 2020-2023  润新知