泛型中super和extends的用法及区别

泛型是在Java 1.5中被加入了,这里不讨论泛型的细节问题,这个在Thinking in Java第四版中讲的非常清楚,这里要讲的是super和extends关键字,以及在使用这两个关键字的时候为什么会不同的限制。 
   首先,我们定义两个类,A和B,并且假设B继承自A。下面的代码中,定义了几个静态泛型方法,这几个例子随便写的,并不是特别完善,我们主要考量编译失败的问题: 
Java代码   收藏代码
  1. public class Generic{  
  2. //方法一  
  3. public static <T extends A> void get(List<T extends A> list)  
  4. {  
  5.     list.get(0);  
  6. }  
  7.   
  8. //方法二  
  9. public static <T extends A> void set(List<T extends A> list, A a)  
  10. {  
  11.     list.add(a);  
  12. }  
  13.   
  14. //方法三  
  15. public static <T super B> void get(List<T super B> list)  
  16. {  
  17.     list.get(0);  
  18. }  
  19.   
  20. //方法四  
  21. public static <T super B> void set(List<T super B> list, B b)  
  22. {  
  23.     list.add(b);  
  24. }  
  25. }  

  编译之后,我们会发现,方法二和方法三没有办法通过编译。按照Thinking in Java上的说法,super表示下界,而extends表示上界,方法二之所以没有办法通过,是因为被放到List里面去的可能是A,也可能是任何A的子类,所以编译器没有办法确保类型安全。而方法三之所以编译失败,则是因为编译器不知道get出来的是B还是B的其他的什么子类,因为set方法四允许在list放入B,也允许在list中放入B的子类,也就没有办法保证类型安全。 
  上面的这段解释听起来可能有点奇怪,都是因为编译器无法判断要获取或者设置的是A和B本身还是A和B的其他的子类才导致的失败。那么Java为什么不干脆用一个关键字来搞定呢? 
  如果从下面的角度来解释,就能把这个为什么编译会出错的问题解释的更加的直白和清除,也让人更容易理解,先看下面的代码,还是A和B两个类,B继承自A: 
Java代码   收藏代码
  1. public class Generic2{  
  2.    public static void main(String[] args){  
  3.       List<? extends A> list1 = new ArrayList<A>();  
  4.       List<? extends A> list2 = new ArrayList<B>();  
  5.       List<? super B> list3 = new ArrayList<B>();  
  6.       List<? super B> list4 = new ArrayList<A>();  
  7.    }  
  8. }  

   从上面这段创建List的代码我们就更加容易理解super和extends关键字的含义了。首先要说明的一点是,Java强制在创建对象的时候必须给类型参数制定具体的类型,不能使用通配符,也就是说new ArrayList<? extends A>(),new ArrayList<?>()这种形式的初始化语句是不允许的。 
   从上面main函数的第一行和第二行,我们可以理解extends的含义,在创建ArrayList的时候,我们可以指定A或者B作为具体的类型,也就是,如果<? extends X>,那么在创建实例的时候,我们就可以用X或者扩展自X的类为泛型参数来作为具体的类型,也可以理解为给?号指定具体类型,这就是extends的含义。 
   同样的,第三行和第四行就说明,如果<? super X>,那么在创建实例的时候,我们可以指定X或者X的任何的超类来作为泛型参数的具体类型。 
   当我们使用List<? extends X>这种形式的时候,调用List的add方法会导致编译失败,因为我们在创建具体实例的时候,可能是使用了X也可能使用了X的子类,而这个信息编译器是没有办法知道的,同时,对于ArrayList<T>来说,只能放一种类型的对象。这就是问题的本质。而对于get方法来说,由于我们是通过X或者X的子类来创建实例的,而用超类来引用子类在Java中试合法的,所以,通过get方法能够拿到一个X类型的引用,当然这个引用可以指向X也可以指向X的任何子类。 
   而当我们使用List<? super X>这种形式的时候,调用List的get方法会失败。因为我们在创建实例的时候,可能用了X也可能是X的某一个超类,那么当调用get的时候,编译器是无法准确知晓的。而调用add方法正好相反,由于我们使用X或者X的超类来创建的实例,那么向这个List中加入X或者X的子类肯定是没有问题的,因为超类的引用是可以指向子类的。 
  最后还有一点,这两个关键字的出现都是因为Java中的泛型没有协变特性的倒置的。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java泛型的extend和super是用于限制泛型类型参数的关键字。 extend用于限制泛型类型参数的上界,表示泛型类型参数必须是指定类型的子类或实现类。例如,List<? extends Number>表示泛型类型参数必须是Number类或其子类。 super用于限制泛型类型参数的下界,表示泛型类型参数必须是指定类型的父类或超类。例如,List<? super Integer>表示泛型类型参数必须是Integer类或其父类。 使用extend和super可以使泛型类型参数更加灵活,可以适应不同的场景需求。但是需要注意的是,使用过多的extend和super可能会导致代码可读性降低,因此需要根据实际情况进行选择和使用。 ### 回答2: Java可以通过泛型实现更强大的类型安全和代码的复用性。在使用泛型时,我们常常需要了解泛型的extend和super关键字,以便更好地理解和使用泛型的extend关键字 泛型的extend关键字用于限定泛型类型参数的上限,表示泛型类型参数必须是继承了某个类或实现了某个接口才能被使用。例如: ``` public class GenericClass<T extends Number> { private T data; public GenericClass(T data) { this.data = data; } public T getData() { return data; } } ``` 在上面的泛型使用了T extends Number语法,表示泛型类型参数T必须是继承了Number类的某个子类,否则编译会报错。这样可以保证data属性的类型一定是Number或其子类类型。 泛型super关键字 泛型super关键字用于限定泛型类型参数的下限,表示泛型类型参数必须是某个类的父类或某个接口的实现类。例如: ``` public class GenericClass<T super Number> { private T data; public GenericClass(T data) { this.data = data; } public T getData() { return data; } } ``` 在上面的泛型使用了T super Number语法,表示泛型类型参数T必须是Number类的父类,即可以是Number类本身或其父类类型。这样可以保证data属性的类型一定是Number类或其父类类型。 需要注意的是,泛型的extend和super关键字只能用于泛型类型参数,而不能用于普通类型。在实际使用,我们需要根据具体的情况来灵活地使用泛型的extend和super关键字,以便在代码实现更好的类型安全和更高的代码复用性。 ### 回答3: Java泛型是一种非常强大的语言特性,允许我们使用一种类型来表示多种类型,在编写类型安全且可重用的代码时经常用到。在泛型使用,我们有时需要约束所能使用泛型类型,使得代码更加类型安全。在这种情况下,我们可以使用泛型的 extend 和 super 关键字。 在 Java 使用 extends 关键字可以为泛型类型参数添加上限限制。通过 extends 关键字,我们可以指定泛型类型参数必须是某个类或接口的子类。这样做的好处是能够防止代码使用无效的类型参数,从而保证代码的类型安全性。值得注意的是,我们只能使用 extends 关键字来添加上限限制,而不能添加下限限制。 下面是一个例子,演示如何使用 extends 关键字: ``` public class Box<T extends Number> { private T t; public Box(T t) { this.t = t; } public T get() { return t; } public void set(T t) { this.t = t; } } ``` 可以看到,Box 类使用extends 关键字,将泛型类型参数 T 限制为 Number 类型或其子类型。在此之后,我们就可以在 Box 类安全地使用 Number 类型或其子类型,如 Integer、Double 等。 除了上限限制之外,Java 还提供了另一种泛型类型参数的约束方式:super 关键字。使用 super 关键字,我们可以限制泛型类型参数必须是某个类或接口的父类。这个约束通常用于需要消费泛型对象的情况,例如集合类的 add 方法。通过为泛型类型参数添加 super 关键字,我们可以保证只能添加某个类的子类到集合,从而防止集合出现无效的元素。类似地,我们也只能使用 super 关键字添加下限限制,而不能添加上限限制。 这里是一个例子,演示如何使用 super 关键字: ``` public class Box<T> { private List<? super T> list; public Box() { list = new ArrayList<>(); } public void add(T t) { list.add(t); } public void addAll(List<? extends T> otherList) { list.addAll(otherList); } } ``` 可以看到,Box 类使用super 关键字来限定集合存储的类型,只能是 T 的父类或其本身。这样做的好处是,保证添加到集合的元素都是类型安全的,避免了程序运行时出现错误。 总之,泛型类型参数的 extend 和 super 关键字是 Java 泛型使用的重要工具,能够帮助我们实现类型安全的代码。在实际使用,根据具体场景灵活运用这两个关键字,能够提高代码的可读性和可维护性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值