Java中的逆变与协变

看下面一段代码:

// public final class Integer extends Number
Number num = new Integer(1);  
List<Number> list = new ArrayList<>();
list.add(new Integer(3));
ArrayList<Number> list = new ArrayList<Integer>(); //type mismatch
 
List<? extends Number> list = new ArrayList<Number>();
list.add(new Integer(1)); //error

为什么Number的对象可以由Integer实例化,而ArrayList<Number>的对象却不能由ArrayList<Integer>实例化?list中的<? extends Number>声明其元素是Number或Number的派生类,为什么不能add Integer?为了解决这些问题,需要了解Java中的逆变和协变以及泛型中通配符用法。
 

1. 逆变与协变

Java中String类型是继承自Object的,姑且记做String ≦ Object,表示String是Object的子类型,String的对象可以赋给Object的对象。而Object的数组类型Object[],理解成是由Object构造出来的一种新的类型,可以认为是一种构造类型,记f(Object),那么可以这么来描述协变和逆变:
       当A ≦ B时,如果有f(A) ≦ f(B),那么f叫做协变;
       当A ≦ B时,如果有f(B) ≦ f(A),那么f叫做逆变;
       如果上面两种关系都不成立则叫做不可变。

2. 泛型中的通配符实现协变与逆变

JAVA中泛型是不变的,可有时需要实现逆变与协变,怎么办呢?这时就需要通配符?。
       <? extends>实现了泛型的协变,比如:

List<? extends Number> list = new ArrayList<>();

 

“? extends Number”则表示通配符”?”的上界为Number,换句话说就是,“? extends Number”可以代表Number或其子类,但代表不了Number的父类(如Object),因为通配符的上界是Number。
       于是有“? extends Number” ≦ Number,则List<? extends Number> ≦ List< Number >。那么就有:


List<? extends Number> list001 = new ArrayList<Integer>();
List<? extends Number> list002 = new ArrayList<Float>();

但是这里不能向list001、list002添加除null以外的任意对象。可以这样理解一下,List<Integer>可以添加Interger及其子类,List<Float>可以添加Float及其子类,List<Integer>、List<Float>都是List<? extends Animal>的子类型,如果能将Float的子类添加到List<? extends Animal>中,就说明Float的子类也是可以添加到List<Integer>中的,显然是不可行。故java为了保护其类型一致,禁止向List<? extends Number>添加任意对象,不过却可以添加null。

       <? super>实现了泛型的逆变,比如:

List<? super Number> list = new ArrayList<>();

“? super Number” 则表示通配符”?”的下界为Number。为了保护类型的一致性,因为“? super Number”可以是Object或其他Number的父类,因无法确定其类型,也就不能往List<? super Number >添加Number的任意父类对象。但是可以向List<? super Number >添加Number及其子类。

List<? super Number> list001 = new ArrayList<Number>();
List<? super Number> list002 = new ArrayList<Object>();
list001.add(new Integer(3));
list002.add(new Integer(3));

3.PECS

现在问题来了:究竟什么时候用extends什么时候用super呢?《Effective Java》给出了答案:
       PECS: producer-extends, consumer-super.
       比如,一个简单的Stack API:

public class Stack<E>{
    public Stack();
    public void push(E e):
    public E pop();
    public boolean isEmpty();
}

要实现pushAll(Iterable<E> src)方法,将src的元素逐一入栈:

public void pushAll(Iterable<E> src){
    for(E e : src)
        push(e)
}

假设有一个实例化Stack<Number>的对象stack,src有Iterable<Integer>与 Iterable<Float>;在调用pushAll方法时会发生type mismatch错误,因为Java中泛型是不可变的,Iterable<Integer>与 Iterable<Float>都不是Iterable<Number>的子类型。因此,应改为:

// Wildcard type for parameter that serves as an E producer
public void pushAll(Iterable<? extends E> src) {
    for (E e : src)
        push(e);
}

 要实现popAll(Collection<E> dst)方法,将Stack中的元素依次取出add到dst中,如果不用通配符实现:

// popAll method without wildcard type - deficient!
public void popAll(Collection<E> dst) {
    while (!isEmpty())
        dst.add(pop());   
}

 同样地,假设有一个实例化Stack<Number>的对象stack,dst为Collection<Object>;调用popAll方法是会发生type mismatch错误,因为Collection<Object>不是Collection<Number>的子类型。因而,应改为:

// Wildcard type for parameter that serves as an E consumer
public void popAll(Collection<? super E> dst) {
    while (!isEmpty())
        dst.add(pop());
}

 在上述例子中,在调用pushAll方法时生产了E 实例(produces E instances),在调用popAll方法时dst消费了E 实例(consumes E instances)。Naftalin与Wadler将PECS称为Get and Put Principle。

       java.util.Collections的copy方法(JDK1.7)完美地诠释了PECS:

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

PECS总结:

  • 要从泛型类取数据时,用extends;
  • 要往泛型类写数据时,用super;
  • 既要取又要写,就不用通配符(即extends与super都不用)。
### Java 泛型中的逆变 #### 逆变 (Contravariance) 在Java泛型中,逆变是指允许使用某个泛型类型的超类型来替代该泛型类型。具体来说,`? super T` 表示可以接受 `T` 类型或其任何超类的实例。这一特性主要应用于写入操作,确保能够向集合中添加合适的元素。 例如: ```java List<? super Integer> list = new ArrayList<>(); list.add(1); // 合法, 因为Integer是Number的一个子类 // list.get(0) 返回 Object 类型,因为无法确定确切类型 ``` 上述代码展示了如何利用逆变机制实现更灵活的数据存储方式[^1]。 #### (Covariance) 相对于逆变而言,则是指当一个泛型类型作为另一个泛型类型的子类型时所表现出的行为模式。即对于两个不同的实际参数化类型A<T1>, B<T2>, 如果存在继承关系使得T1是T2的子类,则可以说A<T1>也是B<T2>的一种特例形式。在Java里我们通常看到的形式就是`<?> extends T`, 它意味着此位置上的类型量既可以是具体的`T`也可以是从`T`派生出来的任意子类。 下面是一个简单的例子说明的应用场景: ```java public class Animal {} public class Dog extends Animal {} List<Dog> dogs = Arrays.asList(new Dog()); List<? extends Animal> animals = dogs; for (Animal animal : animals){ System.out.println(animal); } ``` 这里可以看到由于采用了的方式定义列表animals, 所以可以直接赋值给由Dog组成的列表dogs而不必担心编译错误的发生[^3]. #### 不性(Invariance) 值得注意的是,在某些情况下,既不支持也不支持逆变的情况被称为不性。这意味着只有当两个泛型的具体类型完全一致的情况下才能互相转换。比如`ArrayList<String>`就不能被当作`ArrayList<Object>`处理,即使String确实是Object的子类之一。 总结起来,通过合理运用这些概念可以帮助开发者编写更加通用且安全高效的程序逻辑结构。同时也能有效减少不必要的强制转型所带来的风险以及提高代码可读性维护便利度。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值