在java中,〈?extends A 〉与〈T extends A〉有什么区别?

2024-11-20 12:46:12
推荐回答(3个)
回答(1):

这是泛型嘛,泛型简单的意思就是说,你不知道你想要的这个类具体是啥,但是你可以知道这个类的相关子类或者父类


所以从上面的观点来看,是没啥区别的

例如下面这个两个方法表达的意思是一样的,都表示参数是一个集合,这个集合可能包含着A或者A的任何子类

public void someMethod(List list);
public void someMethod(List list);

这是一种情况,但是相比而言,一个类中,多处方法都需要这个泛型的时候,要比 方便的多,就像在代码里,你声明了一个变量后,你就可以在这个橘册变量的代码块里任何地方调用,同理就像是声明了一个泛型变量T,这个T是一个A或者A的子类,然后这个变量T在所能圆缺宏用的范围之内,你都可以直接用T表示,不用再写或者,比如:


// 前面定义了T,后面参数就可以用T表示了
public void some(List t);

上面是方法里,这个范围比较窄,放在类里,效扮棚果更明显,比如:

public class B{
    // 一个A类或者A子类的变量
    private T a;
    
    // 一个方法
    public void some(List list);
}

所以可以看到,差别也不太大

回答(2):

经常发现有List< super T>、Set的声明,是什么意思呢?< super T>表示包括T在内的任何T的父类,< extends T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。
extends
List< extends Number> foo3的通配符声明,意味着以下的赋值是合法的: // Number "extends" Number (in this context) List< extends Number> foo3 = new ArrayList< extends Number>(); // Integer extends Number List< extends Number> foo3 = new ArrayList< extends Integer>(); // Double extends Number List< extends Number> foo3 = new ArrayList< extends Double>();

读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你可以读取到Number,因为以上的列表要么包含 Number元素,要么包含Number的类元素。你不能保证读取到Integer,因为foo3可能指向的是List。你 不能保证读取到Double,因为foo3可能指向的是List
写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你不能插入一个Integer元素,因为foo3可能指向 List。你不能插入一个Double元素,因为foo3可能指向List。你不能插入一个 Number元素,因为foo3可能指向List。你不能往List< extends T>中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。唯一可以保证的是,你可以从中读 取到T或者T的子类。
super
现在考虑一下List< super T>。

List<轿森 super Integer> foo3的通配符声明,意味着以下赋值是合法的: // Integer is a "superclass" of Integer (in this context) List< super Integer> foo3 = new ArrayList(); // Number is a superclass of Integer List< super Integer> foo3 = new ArrayList(); // Object is a superclass of Integer List< super Integer> foo3 = new ArrayList();

读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你不能保证读取到Integer,因为foo3可能指向 List或者List。你不能保证读取到Number,因为foo3可能指向 List。唯一可以保证的是,你可以读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。
写乱汪入操作通过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你可以插入Integer对象,因为上述声明的列表都支持 Integer。你可以插入Integer的子类的对象,因为哗帆仔Integer的子类同时也是Integer,原因同上。你不能插入Double对象,因为 foo3可能指向ArrayList。你不能插入Number对象,因为foo3可能指向 ArrayList。你不能插入Object对象,因为foo3可能指向 ArrayList
PECS
请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。
生产者使用extends
如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成< extends T>,比如List< extends Integer>,因此你不能往该列表中添加任何元素。
消费者使用super
如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表声明成< super T>,比如List< super Integer>,因此你不能保证从中读取到的元素的类型。
即是生产者,也是消费者
如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List
例子
请参考java.util.Collections里的copy方法(JDK1.7):

我们可以从Java开发团队的代码中获得到一些启发,copy方法中使用到了PECS原则,实现了对参数的保护。

回答(3):

可以说没什么区别