java 泛型

作者: GeorgeDon | 来源:发表于2018-01-29 23:59 被阅读0次

本文以list为例,其他集合类似。
为了安全,编译器在List进行存储的时候只会保存为一种类型。

List<? super T> list中的元素全是T的某个父类型
List<? extends T> list中的元素全是T的某个子类类型

上面这种定义类型的使用场景:

PECS指“Producer Extends,Consumer Super”。换句话说,如果参数化类型表示一个生产者,就使用<? extends T>;如果它表示一个消费者,就使用<? super T>。

在实际的使用中,<? extends T> 实际上代表的是T的某个子类<X>,X继承自T,当我们插入一个数据T的时候无法保证其能够转换成X,但是我们可以保证读取的数据可以转换成T。
同理<? super T>实际上也是代表的T的某个父类<Y>,当我们读取一个数据的时候是无法保障Y能够转换成T的,但是能够保障插入T数据能够转换成Y。

由于List<? extends T>是list中的某个子类,如果我们向list中添加元素,子类之间并不能相互转换,导致添加失败,而List<? super T> 则可以用来添加T的子类,因为T的子类型都可以转为T的某个父类。 相反List<? extends T>中的元素则都可以转为T,则可以用来读取list中的元素,但是List<? super T> 中的元素不能转为T则无法用来读取list中的元素,除非读取为Object 类型。
这样这两个类型的用途就很明显:

List<? super T> 保证向list中添加的元素全是T的子类含T,用于只写操作,读操作可能会导致异常
List<? extends T> 保证读取list中的元素全是T的子类含T ,用于只读操作,写操作会有异常

用例(java.util.Collections的copy方法):

    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());
            }
        }
    }

保证dest只用来存储数据,src的数据只会被读取。
附加:
对于实现了RandomAccess接口的列表,随机读取的效率高于迭代器,而没有实现RandomAccess的接口的列表使用迭代器效率要高一些。(Java接口RandomAccess全面了解

参考
Java 泛型详解

相关文章

网友评论

    本文标题:java 泛型

    本文链接:https://www.haomeiwen.com/subject/pcgfzxtx.html