第23条:请不要在新代码中使用原生态类型
类型参数:List<E>中的E就是类型参数,表示列表中的元素类型
参数化类型:List<String>
原生态类型:List
使用原生态类型的缺点:
当我们获取一个值的时候,必须进行强制类型转换。
如果放入了非预期的类型,编译时我们不会收到任何的错误提示,运行时java.lang.ClassCastException
原生态类型和参数化类型的区别:
比如List和List<String>,前者是逃避了泛型检查,后者则明确告诉编译器,它能够持有任意类型的对象。
泛型有子类型化的规则,List<String>是原生态类型List的子类型,而不是参数化类型List<Object>的子类型。
无限制的通配符类型 如果要使用泛型,但不确定或者不关心实际的参数类型,就可以使用问号代替。Set<?>
无限制通配类型Set<?>和原生态类型Set之间的区别:
可以将任何元素放进使用原生态类型的集合中
不能将任何元素放到无限制通配类型中(除了null)。
不要在新代码中使用原生态类型:(原因就是泛型信息在运行时被擦除)
在类文字中必须使用原生态类型。
instanceof操作符后面的类型不能是泛型。
第24条:消除非受检警告
非受检警告很重要,不要忽略它们。
如果无法消除非受检警告,同时可以证明引起警告的代码是类型安全的,就可以在尽可能小的范围中,用@SuppressWarnings("uncheck")注解禁止该警告。
第25条:列表优先于数组
数组与泛型
数组是协变的,就是如果Sub为Super的子类型,那么数组类型Sub[]就是Super[]的子类型。
泛型是不可变的,对于任意两个不同类型Type1和Type2,List<Type1>既不是List<Type2>的子类型,也不是List<Type2>的父类型。
数组是具体化的,数组是在运行时才知道并检查他们的元素类型约束;泛型只是在编译时强化它们的类型信息。
Object[] objectArray = new String[10];
objectArray[0] = "100";
objectArray[1] = 100;//java.lang.ArrayStoreException
List<String> strlist = new ArrayList<String>();
List<Integer> intlist = new ArrayList<Integer>();
System.out.println(strlist.getClass() == intlist.getClass());
//true,因为类型擦除,可以查看字节码得知
数组和泛型不能混合使用。
用列表代替数组
第26条:优先考虑泛型
泛型更加安全
第27条:优先考虑泛型方法
static <E> List<E> asList(E[] a)
编译器会进行类型推导获取E的值。
利用类型推导简化代码:
Map<String,List<String>> anagrams = new HashMap<String, List<String>>();
//静态泛型工厂
public static <K,V> HashMap<K,V> newHashMap(){
return new HashMap<K,V>();
}
Map<String, List<String>> = newHashMap();
第28条:利用有限通配符来提升API的灵活性
每个类型都是自身的子类型,所以有限通配符也可以把自身类型传进去。
生产者:< ? extends T>
消费者:<? super T>
producer-extends,consumer-super
不要用通配符类型作为返回类型
类型推导并非总能完成工作:
Set<Integer> integers = ...;
Set<Double> doubles = ...;
Set<Number> numbers = union(integers, doubles);//incompatible types
//显式的类型参数
Set<Number> numbers = Union.<Number>union(integers, doubles);
public static E void swap(List<E> list, int i, int j);
public static void swap(List<?> list, int i, int j);
如果类型参数只在方法声明中出现一次,就可以使用通配符取代它。
网友评论