1.Java如果没有泛型会有什么灾难?
-泛型实现类型参数化,将类型检查从运行期提前到编译期,限定元素类型,避免强制类型转换出现类型转换失败的异常(java.lang.ClassCastException)。但编译后字节码可以看出类型参数只有Object;
-编码阶段显式看出对象类型,提高代码可读性;
-采用通配符可以提高代码重用。
代码示例:用上泛型,添加类型不匹配的元素在编译期就会报错。
List<String> list = new ArrayList<>();
list.add("easy code");
//取数据时不需要强制类型转换,避免了出现类型转换失败的异常
String str = list.get(0);
Integer num = 100;
//限制添加类型为String,添加Integer类型变量编译时就会提示不允许添加
list.add(num);
2. List<? extends T>和List<? super T>有哪些区别?
-List<? extends T>设置通配符上限。其中T可以是类或接口,类型参数是T类型或要实现接口或是子类,extends后可以是多个接口(逗号分隔),类只能放第一个位置。List<? extends T>只支持作为T类型get,由于子类类型未知,无法添加null之外的任何对象,null是任何引用类型的实例。
List<? extends Number> listExtendsNumber = new ArrayList();
List<Integer> listInteger = new ArrayList();
listInteger.add(1);
listInteger.add(2);
listExtendsNumber = listInteger;
// 支持作为Number类型取数据
Number i = listExtendsNumber.get(1);
Integer ii = 10;
// 编译错误,无法添加null之外的任何对象
listExtendsNumber.add(ii);
// 可以添加null
listExtendsNumber.add(null);
-List<? super T>设置通配符下限。其中类型参数需要是T类型或是T的超类。List<? super T>只支持添加数据,提取数据时只能作为Object。
List<? super Integer> listSuperInteger = new ArrayList();
listSuperInteger.add(20);
listSuperInteger.add(10);
// 编译错误,List<? super Integer> 取数据只能用Object来接
Integer i = listSuperInteger.get(1);
3. 类名<? super T>存在哪些实际应用场景?(参考Comparator)
查看jdk doc,发现挺多方法都用到类名<? super T>,如TreeSet(Comparator<? super E> comparator),List的sort方法sort(Comparator<? super E> c)可以采用E类型的比较器,或其超类的比较器;Collections的copy方法,copy(List<? super T> dest, List<? extends T> src)