通配符<? extends T>和<? super T>区别及使用场景

<? extends T> : 上界通配符(Upper Bounds Wildcards)
<? super T> : 下界通配符(Lower Bounds Wildcards)

平时开发会遇到通配符<? extends T>和<? super T>的用法,比如:Collections类提供的集合元素拷贝的copy()方法就同时使用了<? extends T>和<? super T>。

public class Collections {

	public static <T> void copy(List<? super T> dest, List<? extends T> src) {
        int srcSize = src.size();
        if (srcSize > dest.size())
            throw new IndexOutOfBoundsException("Source does not fit in dest");

        if (srcSize < COPY_THRESHOLD ||
            (src instanceof RandomAccess && dest instanceof RandomAccess)) {
            for (int i=0; i<srcSize; i++)
                dest.set(i, src.get(i));
        } else {
            ListIterator<? super T> di=dest.listIterator();
            ListIterator<? extends T> si=src.listIterator();
            for (int i=0; i<srcSize; i++) {
                di.next();
                di.set(si.next());
            }
        }
    }

}

baomidou的分页查询page()方法使用了<? extends T>:

<E extends IPage<T>> E page(E page, Wrapper<T> queryWrapper);

但为什么要这么用?如果我们使用的话可以用在哪些场景?下面我们通过小示例来理解它的作用和使用场景。


我们发现,类的继承关系能做到对子类型的兼容,但泛型没办法对子类型兼容,编译期就会报错。
在这里插入图片描述

在这里插入图片描述
解决编译错误,可以使用到通配符,满足泛型对子类型的兼容。
在这里插入图片描述
除了<? extends T> 用法,还有另一种用法<? super T>。下面我们来看看他们的区别。

声明包含上下界通配符的集合类型:
在这里插入图片描述
使包含上下界通配符的集合类型:
<? extends Number>

  • add() : 编译器只知道元素类型是 Number 或 Number 的子类,所以有可能是 Number、Integer、Long、Float、Double其中一个类型,为保证类型安全不能添加除了 null 以外的任何元素,即使是 Number 本身也不行。
  • get() : 既然编译器不知道此时集合中的元素是 Integer、Long、Float、Double 的哪一个,返回类型只能是他们共同父类 Number 类型。

<? super Number>

  • add() : 编译器只知道元素类型是 Number 或Number 的父类,所以有可能是Number、Object其中一个。编译器知道下界是 Number ,根据类型向上兼容所以可以添加的元素是 Number 以及 Number 的子类。
  • get() : 既然编译器不确定集合类型是 Number 或Number父类的哪一种类型,add()的又是哪类型,返回类型只能是他们的共同父类 Object 。
    在这里插入图片描述
    特点:<? extends T> 的 add() 被限制,<? super T> 的 get() 被限制。

所以,<? extends T> 上界通配符,被设计为用来读取数据的泛型(只能读取数据为T的元素);<? super T> 下界通配符,被设计为用来写数据的泛型(只能写入T或T的子类型)。

回头看看文章开头提到的两个使用场景,是不是就明白了。



示例源码:

public static void main(String[] args) {

        /**
         * 上界通配符,通配符?只能是 T 和 T 的子类
         * add(): 不可以加入任何元素,除了null
         * get():返回的值的类型默认是T
         */
        ArrayList<? extends Number> list1 = new ArrayList<Integer>(); //子类
        list1.add(1);    //编辑报错,因为通配符?可能是T或T的子类
        list1.add(null); //null可以,因为null可以为任何类型,但无意义

        Number number = list1.get(0); //默认是T类型,从声明泛型集合时我们知道,可以声明T和T的子类任何一个,所以返回默认只能是T,如果是子类再强转
        Integer integer = (Integer) list1.get(0); //可以强转成子类型,但如果不是Integer会包运行错误java.lang.ClassCastException

        /**
         * 下界通配符,通配符?只能是 T 和 T 的父类
         * add(): 只能添加类型为 T 和 T 的子类
         * get():返回的值默认是 Object类型
         */
        ArrayList<? super Number> list2 = new ArrayList<Number>();
        list2.add(1);
        list2.add(1L);
        list2.add(1.1f);
        list2.add(2.5d);

        Object object = list2.get(0); //默认是Object类型,因为在add()时可以是T和T的子类,所以返回默认是Object
}
  • 7
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java的泛型中,`<? extends T>`和`<? super T>`是用来限定通配符(Wildcard)的上界和下界。 1. `<? extends T>`:表示通配符的上界是T或T的子类。使用`<? extends T>`可以使泛型类型接受T或T的子类型作为参数,但不能用于写入对象。 ```java public void processList(List<? extends Number> list) { for (Number num : list) { System.out.println(num); } } List<Integer> integers = new ArrayList<>(); integers.add(1); integers.add(2); processList(integers); // 可以传递List<Integer>或List<Number>,但不能传递List<Object> ``` 在上述示例中,`processList()`方法接受一个`List<? extends Number>`类型的参数,这意味着可以传递`List<Integer>`或`List<Number>`作为参数。在方法内部,我们可以从list中读取Number类型的元素,因为Number是Integer的父类。 2. `<? super T>`:表示通配符的下界是T或T的父类。使用`<? super T>`可以使泛型类型接受T或T的父类型作为参数,并且可以用于写入对象。 ```java public void addToList(List<? super Integer> list) { list.add(1); list.add(2); } List<Number> numbers = new ArrayList<>(); numbers.add(0.5); addToList(numbers); // 可以传递List<Integer>或List<Object>,但不能传递List<Number> System.out.println(numbers); // 输出:[0.5, 1, 2] ``` 在上述示例中,`addToList()`方法接受一个`List<? super Integer>`类型的参数,这意味着可以传递`List<Integer>`或`List<Object>`作为参数。在方法内部,我们可以向list中添加Integer类型的元素,因为Integer是Number的子类。 总结: - `<? extends T>`用于限定泛型的上界,可以读取泛型对象,但不能写入; - `<? super T>`用于限定泛型的下界,可以写入泛型对象,但读取时需要进行类型转换。 使用通配符的目的是为了增加泛型的灵活性,在不确定具体类型的情况下,能够处理更广泛的数据类型。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值