基础:
extend 关键字,使用关键字extend继承父类的所有公共方法与公共字段。对于:
● 构造方法不能被继承,子类要重写
● 父类的私有属性不能被继承。
● 使用final声明的类是最终类,也不能被继承
super关键字来实现对父类成员的访问,用来引用当前对象的父类
一个是继承,一个是引用。
常见符号:
E - Element(在集合中使用,因为集合中存放的是元素)
T - Type (Java类)
K - Key(键)
V - Value(值)
N - Number(数值类型)
? - 表示不确定的java类型(无限制通配符类型)
PECS原则
假设现在有这么一个类的继承树,Fruit -> Apple -> RedApple/GreenApple
类:
Apple
public class Apple extends Fruit {
private Double price;
public Apple(Double price) {
this.price = price;
this.show();
}
public Double getPrice() {
return price;
}
@Override
public void show() {
System.out.println("I am an app");
}
}
GreenApple
public class GreenApple extends Apple {
public GreenApple(Double price) {
super(price);
}
@Override
public void show() {
System.out.println("I am a green apple");
}
}
RedApple
public class RedApple extends Apple {
public RedApple(Double price) {
super(price);
}
@Override
public void show() {
System.out.println("I am a red apple");
}
}
测试
List<? extends Fruit> 意思: List中所有元素都是Fruit的子类(包含本身), List<? super Fruit> 意思: List中所有元素都是Fruit的父类(包含本身)
public static void main(String[] args) {
// 申明父类,使用extend的时候,里面的元素都是Fruit的子类,无法识别
// List <? extends Fruit> appList2 = new ArrayList<>();
// appList2.add(new Apple(6.0));
// appList2.add(new RedApple(8.5));
List <? super Fruit> appList3 = new ArrayList<>();
appList3.add(new Apple(6.0));
appList3.add(new RedApple(7.5));
appList3.add(new GreenApple(4.0));
System.out.println(JSON.toJSONString(appList3));
Object object = appList3.get(1);
System.out.println(JSON.toJSONString(object));
}
“?”不能添加元素
/*
1. “?”不能添加元素
以“?”声明的集合,不能往此集合中添加元素,只能读,所以它只能作为生产者(亦即它只能被迭代),如下:
*/
public static void testGeneric() {
List<?> names = Lists.newArrayList("aaa", "yan");
// names.add("ccc"); // 报错无法进行添加
// 通配符声明的集合,获取的元素都是Object类型
List<Object> allNames = Lists.newArrayList("aaaa");
allNames.addAll(names);
// 只能以Object迭代元素
for (Object name : allNames) {
System.out.println(name);
}
System.out.println("==============testGeneric===============");
}
“? extends T”也不能添加元素
/*
“? extends T”也不能添加元素
以“? extends T”声明的集合,不能往此集合中添加元素,所以它也只能作为生产者
*/
public static void testGenericExtend() {
List<? extends String> names = Lists.newArrayList("bbb");
// names.add("yan"); // 报错,无法进行添加
// 声明消费者
List<String> allNames = Lists.newArrayList("bbbb");
// 消费生产者的元素
allNames.addAll(names);
allNames.forEach(System.out::println);
System.out.println("==============testGenericExtend===============");
}
“? super T”能添加元素
/*
“? super T”能添加元素
在通配符的表达式中,只有“? super T”能添加元素,所以它能作为消费者(消费其他通配符集合)。
*/
public static void testGenericSuper() {
List<? super String> allNames = Lists.newArrayList("bbb");
// 可以直接添加本类及其父元素
allNames.add("yan");
// 可以直接添加泛型元素
List<String> names = Lists.newArrayList("bbbb");
// 也可以添加通配符泛型元素
allNames.addAll(names);
List<? extends String> names1 = Lists.newArrayList("cccc");
allNames.addAll(names1);
// 只能以Object迭代元素,因为是Object类型,就难以继续其它操作
for (Object name : allNames) {
System.out.println(name);
}
System.out.println("==============testGenericSuper===============");
}
总结:
? extends T”通配符,消费者使用“? super T”通配符
总结PECS原则如下:
如果要从集合中读取类型T的数据,并且不能写入,可以使用 ? extends 通配符;(Producer Extends)
如果要从集合中写入类型T的数据,并且不需要读取,可以使用 ? super 通配符;(Consumer Super)
如果既要存又要取,那么就不要使用任何通配符。