面试题整理:extends 和super 泛型限定符

请解释一下extends 和super 泛型限定符

上界不存下界不取
extends上限通配符,用来限制类型的上限,只能传入本类和子类,add方法受阻,可以从一个数据类型里获取数据;
super下限通配符,用来限制类型的下限,只能传入本类和父类,get方法受阻,可以把对象写入一个数据结构里;
在java泛型中,? 表示通配符,代表未知类型,< ? extends Object>表示上边界限定通配符,< ? super Object>表示下边界限定通配符。
只能用于方法返回,告诉编译器此返参的类型的最小继承边界为T,T和T的父类都能接收,但是入参类型无法确定,只能接受null的传入
只能用于限定方法入参,告诉编译器入参只能是T或其子类型,而返参只能用Object类接收
? 既不能用于入参也不能用于返参

通配符 与 T 的区别
T:作用于模板上,用于将数据类型进行参数化,不能用于实例化对象。
?:在实例化对象的时候,不确定泛型参数的具体类型时,可以使用通配符进行对象定义。
< T > 等同于 < T extends Object>
< ? > 等同于 < ? extends Object>

例一:定义泛型类,将key,value的数据类型进行< K, V >参数化,而不可以使用通配符。

public class Container {
    private K key;
    private V value;
 
    public Container(K k, V v) {
        key = k;
        value = v;
    }
}

例二:实例化泛型对象,我们不能够确定eList存储的数据类型是Integer还是Long,因此我们使用List定义变量的类型。

List eList = null;
eList = new ArrayList();
eList = new ArrayList();

上界类型通配符(? extends)

List eList = null;
eList = new ArrayList();
Number numObject = eList.get(0);  //语句1,正确
 
//Type mismatch: cannot convert from capture#3-of ? extends Number to Integer
Integer intObject = eList.get(0);  //语句2,错误
 
//The method add(capture#3-of ? extends Number) in the type List is not applicable for the arguments (Integer)
eList.add(new Integer(1));  //语句3,错误

语句1:ListeList存放Number及其子类的对象,语句1取出Number(或者Number子类)对象直接赋值给Number类型的变量是符合java规范的。
语句2:ListeList存放Number及其子类的对象,语句2取出Number(或者Number子类)对象直接赋值给Integer类型(Number子类)的变量是不符合java规范的。
语句3:ListeList不能够确定实例化对象的具体类型,因此无法add具体对象至列表中,可能的实例化对象如下。

eList = new ArrayList();
eList = new ArrayList();
eList = new ArrayList();

总结:上界类型通配符add方法受限,但可以获取列表中的各种类型的数据,并赋值给父类型(extends Number)的引用。因此如果你想从一个数据类型里获取数据,使用 ? extends 通配符。限定通配符总是包括自己。

下界类型通配符(? super )

List sList = null;
sList = new ArrayList();
 
//Type mismatch: cannot convert from capture#5-of ? super Integer to Number
Number numObj = sList.get(0);  //语句1,错误
 
//Type mismatch: cannot convert from capture#6-of ? super Integer to Integer
Integer intObj = sList.get(0);  //语句2,错误
 
sList.add(new Integer(1));  //语句3,正确

语句1:List 无法确定sList中存放的对象的具体类型,因此sList.get获取的值存在不确定性,子类对象的引用无法赋值给兄弟类的引用,父类对象的引用无法赋值给子类的引用,因此语句错误。
语句2:同语句1。
语句3:子类对象的引用可以赋值给父类对象的引用,因此语句正确。
总结:下界类型通配符get方法受限,但可以往列表中添加各种数据类型的对象。因此如果你想把对象写入一个数据结构里,使用 ? super 通配符。限定通配符总是包括自己。

总结
限定通配符总是包括自己
上界类型通配符:add方法受限
下界类型通配符:get方法受限
如果你想从一个数据类型里获取数据,使用 ? extends 通配符
如果你想把对象写入一个数据结构里,使用 ? super 通配符
如果你既想存,又想取,那就别用通配符
不能同时声明泛型通配符上界和下界
面试题整理:extends 和super 泛型限定符_第1张图片

你可能感兴趣的:(面试题整理:extends 和super 泛型限定符)