泛型中? super T和? extends T的区别

原文出处: 并发编程网java

常常发现有List<? super T>、Set<? extends T>的声明,是什么意思呢?<? super T>表示包括T在内的任何T的父类,<? extends T>表示包括T在内的任何T的子类,下面咱们详细分析一下两种通配符具体的区别。编程

 

extends

List<? extends Number> foo3的通配符声明,意味着如下的赋值是合法的:并发

01 // Number "extends" Number (in this context)
02  
03 List<? extends Number> foo3 = new ArrayList<? extends Number>();
04  
05 // Integer extends Number
06  
07 List<? extends Number> foo3 = new ArrayList<? extends Integer>();
08  
09 // Double extends Number
10  
11 List<? extends Number> foo3 = new ArrayList<? extends Double>();
  1. 读取操做经过以上给定的赋值语句,你必定能从foo3列表中读取到的元素的类型是什么呢?你能够读取到Number,由于以上的列表要么包含Number元素,要么包含Number的类元素。

    你不能保证读取到Integer,由于foo3可能指向的是List<Double>。ide

    你不能保证读取到Double,由于foo3可能指向的是List<Integer>。函数

  2. 写入操做过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?

    你不能插入一个Integer元素,由于foo3可能指向List<Double>。this

    你不能插入一个Double元素,由于foo3可能指向List<Integer>。spa

    你不能插入一个Number元素,由于foo3可能指向List<Integer>。.net

    你不能往List<? extends T>中插入任何类型的对象,由于你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。惟一能够保证的是,你能够从中读取到T或者T的子类。code

super

如今考虑一下List<? super T>。对象

List<? super Integer> foo3的通配符声明,意味着如下赋值是合法的:

01 // Integer is a "superclass" of Integer (in this context)
02  
03 List<? super Integer> foo3 = new ArrayList<Integer>();
04  
05 // Number is a superclass of Integer
06  
07 List<? super Integer> foo3 = new ArrayList<Number>();
08  
09 // Object is a superclass of Integer
10  
11 List<? super Integer> foo3 = new ArrayList<Object>();
  1. 读取操做经过以上给定的赋值语句,你必定能从foo3列表中读取到的元素的类型是什么呢?你不能保证读取到Integer,由于foo3可能指向List<Number>或者List<Object>。

    你不能保证读取到Number,由于foo3可能指向List<Object>。

    惟一能够保证的是,你能够读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。

  2. 写入操做经过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你能够插入Integer对象,由于上述声明的列表都支持Integer。

    你能够插入Integer的子类的对象,由于Integer的子类同时也是Integer,缘由同上。

    你不能插入Double对象,由于foo3可能指向ArrayList<Integer>。

    你不能插入Number对象,由于foo3可能指向ArrayList<Integer>。

    你不能插入Object对象,由于foo3可能指向ArrayList<Integer>。

PECS

请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。

  • 生产者使用extends

若是你须要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你须要把这个列表声明成<? extends T>,好比List<? extends Integer>,所以你不能往该列表中添加任何元素。

  • 消费者使用super

若是须要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你须要把这个列表声明成<? super T>,好比List<? super Integer>,所以你不能保证从中读取到的元素的类型。

  • 便是生产者,也是消费者

若是一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,好比List<Integer>。

例子

请参考java.util.Collections里的copy方法(JDK1.7):

 

引用例子:

 

泛型中使用通配符有两种形式:子类型限定<? extends xxx>和超类型限定<? super xxx>。

(1)子类型限定

下面的代码定义了一个Pair<T>类,以及Employee,Manager和President类。

 
  1. public class Pair<T> {  
  2.     private T first;  
  3.     private T second;  
  4.   
  5.     public Pair(T first, T second) {  
  6.         this.first = first;  
  7.         this.second = second;  
  8.     }  
  9.   
  10.     public T getFirst() {  
  11.         return first;  
  12.     }  
  13.   
  14.     public T getSecond() {  
  15.         return second;  
  16.     }  
  17.   
  18.     public void setFirst(T newValue) {  
  19.         first = newValue;  
  20.     }  
  21.   
  22.     public void setSecond(T newValue) {  
  23.         second = newValue;  
  24.     }  
  25. }  
  26.   
  27. class Employee {  
  28.     private String name;  
  29.     private double salary;  
  30.       
  31.     public Employee(String n, double s) {  
  32.         name = n;  
  33.         salary = s;  
  34.     }  
  35.       
  36.     public String getName() {  
  37.         return name;  
  38.     }  
  39.   
  40.     public double getSalary() {  
  41.         return salary;  
  42.     }  
  43. }  
  44.   
  45. class Manager extends Employee {  
  46.     public Manager(String n, double s) {  
  47.         super(n, s);  
  48.     }  
  49. }  
  50. <pre name="code" class="java">  
  51. class President extends Manager {  
  52.     public President(String n, double s) {  
  53.         super(n, s);  
  54.     }  
  55. }  

 

 
 

如今要定义一个函数能够打印Pair<Employee>


 在CODE上查看代码片派生到个人代码片
  1. public static void printEmployeeBoddies(Pair<Employee> pair) {  
  2.     System.out.println(pair.getFirst().getName() + ":" + pair.getSecond().getName());  
  3. }  

但是有一个问题是这个函数输入参数只能传递类型Pair<Employee>,而不能传递Pair<Manager>和Pair<President>。例以下面的代码会产生编译错误

 
  1. Manager mgr1 = new Manager("Jack", 10000.99);  
  2. Manager mgr2 = new Manager("Tom", 10001.01);  
  3. Pair<Manager> managerPair = new Pair<Manager>(mgr1, mgr2);  
  4. PairAlg.printEmployeeBoddies(managerPair);  

之因此会产生编译错误,是由于Pair<Employee>和Pair<Manager>其实是两种类型。

由上图能够看出,类型Pair<Manager>是类型Pair<? extends Employee>的子类型,因此为了解决这个问题能够把函数定义改为
public static void printEmployeeBoddies(Pair<? extends Employee> pair)

可是使用通配符会不会致使经过Pair<? extends Employee>的引用破坏Pair<Manager>对象呢?例如:
Pair<? extends Employee> employeePair = managePair;employeePair.setFirst(new Employee("Tony", 100));
不用担忧,编译器会产生一个编译错误。Pair<? extends Employee>参数替换后,咱们获得以下代码
? extends Employee getFirst()
void setFirst(? extends Employee)
对于get方法,没问题,由于编译器知道能够把返回对象转换为一个Employee类型。可是对于set方法,编译器没法知道具体的类型,因此会拒绝这个调用。

(2)超类型限定

超类型限定和子类型限定相反,能够给方法提供参数,可是不能使用返回值。? super Manager这个类型限定为Manager的全部超类。

Pair<? super Manager>参数替换后,获得以下方法

? super Manager getFirst()
void setFirst(? super Manager)

编译器能够用Manager的超类型,例如Employee,Object来调用setFirst方法,可是没法调用getFirst,由于不能把Manager的超类引用转换为Manager引用。

超类型限定的存在是为了解决下面一类的问题。例如要写一个函数从Manager[]中找出工资最高和最低的两个,放在一个Pair中返回。

 
  1. public static void minMaxSal(Manager[] mgrs, Pair<? super Manager> pair) {  
  2.     if (mgrs == null || mgrs.length == 0) {  
  3.         return;  
  4.     }  
  5.       
  6.     pair.setFirst(mgrs[0]);  
  7.     pair.setSecond(mgrs[0]);  
  8.     //TODO  
  9. }  

如此就能够这样调用


 在CODE上查看代码片派生到个人代码片
    1. Pair<? super Manager> pair = new Pair<Employee>(null, null);  
    2. minMaxSal(new Manager[] {mgr1, mgr2}, pair);  
相关文章
相关标签/搜索