public class Test_2 { public static void main(String[] args) { List<? extends B> list1 = new ArrayList<>(); // 上界 List<? super B> list2 = new ArrayList<>(); // 下界 A a = new A(); B b = new B(); C c = new C(); Object o = new Object(); // list1.add(o); 不能增加任何元素,因为List中具体是B的哪种子类无奈确定 // list1.add(a); // list1.add(b); // list1.add(c); o = list1.get(0); a = list1.get(0); b = list1.get(0); // c = list1.get(0); 编译谬误,编译器无奈向下转型 // list2.add(o); 编译谬误, // list2.add(a); 因为List中具体是B的哪种父类无奈确定 list2.add(b); list2.add(c); o = list2.get(0); // a = list2.get(0); 编译谬误,因为List中具体是B的哪种父类无奈确定,无奈向下转型,而Object是所有类的父类 // b = list2.get(0); // c = list2.get(0); }}class A {}class B extends A {}class C extends B {}
上界
上界用 extends
关键字申明,示意参数化的类型可能是所指定的类或者其任意子类。例如<? extends B>
,泛型的上界就是 B 类。
形如 List<? extends B>
,具体哪一种不能确定,既能够是 B,也能够是 C。在尝试执行 add() 办法时,List中的类型不能确定是具体哪一种,所以会编译报错。在执行 get() 办法时,不论是 B 还是 C,都能够以 A 类对象来接管。所以 List<? extends B>
不能增加元素,具备只读属性,只能获取。
下界
下界用 super
关键字申明,示意参数化的类型可能是所指定的类型或者其任意父类。例如<? super B>
,泛型的下界就是 B 类。
形如 List<? super B>
,具体哪一种不能确定,既能够是 B,也能够是 A,直至 Object类。在尝试执行 add()
办法时,尽管 List 的具体类型不能确定,然而依据多态, B 类及其子类的对象必定都能够被赋值给 B 的对象,所以只能增加 B 类及其子类的对象。在尝试执行 get()
办法时,List 中的类型是 B 类或者其父类的具体一种,向上直至 Object 类,所以只能将获取的元素赋值给 Object 对象。
以上内容都是我本人通过测试的集体总结,如有纰漏敬请体谅。