1 package com.example.base; 2 3 import java.util.ArrayList; 4 import java.util.List; 5 6 import com.example.spring.MyLog; 7 /** 8 * Producer extends Consumer super 生产者使用extends,consumer使用super 9 * 这里的生产者和消费者是相对容器而言的, 10 * 生产者只能对外提供数据,不可以写入数据,数据来源于赋值操作(将参数化类型为子类的容器赋值过来) 11 * 消费者表示只能向容器中写入数据,不能读取(只能以Object来接收) 12 * 这里的extends和super指的是声明类型和参数化类型的关系, 13 * 如下所示:等号左侧类型为声明类型,右侧为参数化类型 14 * List<? extends Number> intList = new ArrayList<Integer>(); 15 * private List<? super Number> intList2 = new ArrayList<Number>(); 16 * @DESC 17 * @author guchuang 18 * 19 */ 20 public class PECS { 21 22 int int1 = 1; 23 long long1 = 11; 24 Number number1 = 10; 25 26 private List<Integer> intList = new ArrayList<Integer>(); 27 private List<Long> longList = new ArrayList<Long>(); 28 private List<Number> numberList = new ArrayList<Number>(); 29 30 public static void main(String[] args) { 31 PECS pecs = new PECS(); 32 pecs.pe(); 33 pecs.cs(); 34 } 35 36 /** 37 * <? extends Number> 这种方式声明的泛型容器,不能写入任何类型的数据,只能读取数据 38 * 其指定了上确界为Number,参数化类型(如:实际新建的容器类型T new ArrayList<T>())必须是Number的子类,所以读出的数据都可以上转型为Number 39 * 理论含义是容器内包含的数据可能是Number的任何一种子类,所以无法添加数据。 40 * 写数据的方式: 41 * 声明一个具体类型(Number的子类型)的容器,向其中添加数据,将这个容器赋值给numbers容器 42 * 优势: 43 * 避免调用者向此容器内写入数据,只能读取里面的数据 44 */ 45 public void pe() { 46 //<? extends Number> 这种方式声明的泛型容器,不能写入任何类型的数据 47 List<? extends Number> numbers = new ArrayList<Number>(); 48 //下面三种添加数据都会导致编译报错 49 //foo.add(int1); 50 //foo.add(long1); 51 //foo.add(number1); 52 53 //Number number = numbers.get(0); 54 55 intList.add(123); 56 intList.add(456); 57 //将新创建的具有确定类型的容器(必须是Number的子类型)赋值给numbers,起到生产数据的目的 58 numbers = intList; //新容器的参数化类型是Integer,是Number的子类,所以可以赋值 59 MyLog.info(numbers); 60 Number number = numbers.get(0); //读出来的类型依然是Number 61 MyLog.info("read from ? extends Number: " + number); 62 numbers = longList; 63 numbers = numberList; 64 } 65 66 /** 67 * Consumer super 68 * ? super Number 表示容器中所有的数据类型都是Number或者Number的超类型, 69 * 所以Number及其子类型(可以上转型为Number)可以写入,读取的时候由于不能确定类型,只能使用Object接收 70 */ 71 public void cs() { 72 List<? super Number> numbers = new ArrayList<Number>(); 73 numbers.add(int1); 74 numbers.add(long1); 75 numbers.add(number1); 76 //numbers.add(new Object()); 编译报错 77 MyLog.info(numbers); 78 79 //Number n = numbers.get(0); 编译报错 80 Object n = numbers.get(0); //只能以Object来接收数据 81 //numbers = intList; 编译报错 82 numbers = numberList; 83 numbers = new ArrayList<Object>(); 84 } 85 }