提出问题

java开发中如何更好使用范型???

解决问题

以下来自《Effective Java》读书笔记:

请不要在新代码中使用原生态类型
参考高手文章:

http://www.cnblogs.com/nayitian/archive/2013/08/08/3245496.html

http://www.cnblogs.com/TwoWaterLee/p/5878056.html

参数化类型,通配符类型和原生态类型对比:

  • Set是个参数化类型,表示可以包含任何对象的一个集合;
  • Set< ? >则是一个通配符类型,表示只能包含某种未知对象类型的一个集合;
  • Set则是个原生态类型,它脱离了泛型系统。**前两种是安全的,最后一种不安全。

无限制通配类型和原生态类型的区别是:通配符类型是安全的,原生态类型不安全。你可以将任何元素放入到原生态类型的集合中,但不能将除了null之外的其他任何元素放到Collection

消除非受检警告

参考高手文章:
http://blog.csdn.net/lqadam/article/details/52563155

要尽可能地消除每一个非受检的警告。

使用@SuppressWarnings(“unchecked”)注解,需要注意的是将该注解用在尽可能小的范围内,能在变量上使用的不在方法上使用,能在方法上使用的不在类上使用。

使用@SuppressWarnings注解时,都要添加一条注释,说明为什么这么做是类型安全的

列表优先于数组
下面的代码片段是合法的,却是不符合预期:

// 运行时候失败

  1. Object[] objectArray = new Long[1];
  2. objectArray[0] = I dont fit in”; // Throws ArrayStoreException

但下面这段代码则不合法,但却提醒了你怎样才能达到预期:

  1. // Won’t compile!
  2. List<Object> o1 = new ArrayList<Long>(); // Incompatible types
  3. o1.add(“I dont fit in”);

明显,利用列表可以在编译时发现错误。我们当然希望在编译时发现错误了。

总结:数组是协变的,数组是具体化的:在运行时才知道并检查它们的元素类型约束。

优先考虑泛型,优先考虑范型方法

定义泛型方法语法格式如下:

http://www.cnblogs.com/iyangyuan/archive/2013/04/09/3011274.html

参考高手文章:

http://www.cnblogs.com/cutter-point/p/5883279.html

修改前:

  1. //这里使用原生类型是不合理的
  2. public static Set union1(Set s1, Set s2){
  3. Set result = new HashSet(s1);
  4. result.addAll(s2);
  5. return result;
  6. }

修改后:

  1. //这里的范型没有安全警告
  2. public static <E> Set<E> union(Set<E> s1, Set<E> s2){
  3. Set<E> result = new HashSet<E>(s1);
  4. result.addAll(s2);
  5. return result;
  6. }

利用有限制通配符来提升API的灵活性

参考高手文章:

http://www.cnblogs.com/13jhzeng/p/5726511.html

先看下面例子:

修改之前:

  1. public void pushAll(Iterable<E> src) {
  2. for (E e : src) {
  3. push(e)
  4. }
  5. }

修改之后:

  1. //src是生成者,所以使用extend
  2. public void pushAll(Iterable<? extends E> src) {
  3. for (E e : src) {
  4. push(e)
  5. }
  6. }

pushAll的输入参数类型不应该为”E的Iterable接口”,而应该为”E的某个子类型的Iterable接口”,有一个通配符类型正符全此意:Iterable

优先考虑类型安全的异构容器

参考高手文章:

http://blog.csdn.net/zhang_amao/article/details/52107612

http://blog.csdn.net/tkd03072010/article/details/7722110

  1. public Class Favorites{
  2. public <T> void putFavorite(Class<T> type, T instance);
  3. public <T> T getFavorites(Class<T> type);
  4. }

Favorites实例是类型安全的:当你向它请求String的时候,不会返回一个Integer给你。同时它也是异构的:不像普通的map,它的所有的键都是不同类型的。因此,我们将Favorites称作类型安全的异构容器

Map不能保证键和值之间的类型关系,即不能保证每个值的类型都与键的类型相同,当你考虑到这个问题时,可以尝试使用类型安全的异构容器:

  1. public class Favorites {
  2. private Map<Class<?>, Object> favorites =
  3. new HashMap<Class<?>, Object>();
  4. public <T> void putFavorites(Class<T> type, T instance) {
  5. if(type == null)
  6. throw new NullPointerException();
  7. favorites.put(type, type.cast(instance));
  8. }
  9. public <T> T getFavorites(Class<T> type) {
  10. return type.cast(favorites.get(type));
  11. }
  12. }