Java中的泛型--generic

  学习一下Java中的泛型,还是张孝祥老师很好的讲解。。。jdk 1.5中新增的特性

一、泛型初探

  泛型在Java中挺常用的,尤其是项目中也是比较常用的,但是用起来感觉好用,并没有深入的去理解一下,泛型是提供给javac编译器使用的,可以限定集合中的输入类型,让编译器挡住源程序中的非法输入,编译器编译带类型说明的集合时会去掉“类型”信息,使得程序运行不受影响,对于参数化的泛型类型,getClass() 方法的返回值和原始类型完全一样。由于编译生成的字节码会去掉泛型的类型信息,只要能跳过编译器,就可以往某个泛型集合中加入其它类型的数据,例如,用反射得到集合,在调用其add()方法即可。

 1         ArrayList<String> collection1 = new ArrayList<String>();
 2         collection1.add("123");
 3 
 4         ArrayList<Integer> collection2 = new ArrayList<Integer>();
 5         collection2.add(11);
 6 
 7         // 结果为 true 实际编译完成之后 实际上是相同的基本类型
 8         System.out.println(collection1.getClass() == collection2.getClass());
 9         // 利用反射可以跳过编译器
10         collection2.getClass().getMethod("add", Object.class).invoke(collection2, "abc");
11         System.out.println(collection2.get(1));

ArrayList<E> 这种是泛型类型;

E称为泛型变量或者是泛型参数;

ArrayList<Integer>称为参数化的类型;

Integer称为类型参数的实例或者是实际类型参数;

ArrayList为原始类型

二、泛型中的 ? 通配符

总结:使用?通配符可以引用各种参数化的类型,?通配符定义的变量只是用作引用,可以调用与参数化无关的方法,不能调用与参数化有关的方法

通配符的扩展:

扫描二维码关注公众号,回复: 6117265 查看本文章

1) 限定通配符的上边界

正确:Vector<? extends Number> x = new Vector<Integer>();

这种是后面new出来的可以是extends 的子类,例如:Integer 是Number 的子类,是继承关系

错误:Vector<? extends Number> x = new Vector<String>();

2)限定通配符的下边界

正确:Vector<? super Integer> x = new Vector<Number>();

这种是后面new出来的是前面的父类,正好和上面的相反,例如:Number 是Integer的父类

错误:Vector<? super Integer> x = new Vector<Byte>();

三、泛型集合类的综合案例

1   // 简单的map中元素的迭代
2   Map<String,Integer> maps = new HashMap<String,Integer>();
3         maps.put("A", 27);
4         maps.put("B", 28);
5         
6         Set<Map.Entry<String, Integer>> entrySet = maps.entrySet();
7         for(Map.Entry<String, Integer> entry : entrySet){
8             System.out.println(entry.getKey() + ":" + entry.getValue());
9       }

四、自定义泛型方法和应用

1、只有引用类型才能作为泛型方法的实际参数,swap(new int[3]{1,2,3},1,2)语句会报编译错误

2、除了在应用泛型时可以使用extends限定符,在定义泛型的时候也可以使用extends限定符

3、普通方法、构造方法和静态方法中都可以使用泛型

4、在泛型中可以同时有多个类型参数,在定义他们的尖括号中用逗号分,例如:

public static <K,V> v getValue(K key){return map.getKey()};

1     // 交换数组中任意下标中两个元素的位置
2     private static <T> void swap(T[] a, int i, int j){
3         T tmp = a[i];
4         a[i] = a[j];
5         a[j] = tmp;
6     }

5、当一个变量被声明为泛型时,只能被实例变量和方法调用(还有内嵌类型),而不能被静态变量和方法调用,因为静态成员是被所有的参数化的类所共享的,所以静态成员不应该有类级别的类型参数

 1 // 例如baseDao中经常那样定义成泛型类,注意:静态成员没有类级别的类型参数
 2 public class GenericDao <T>{
 3 
 4     public void add(T t){
 5         
 6     }
 7     
 8     public T findById(int id){
 9         return null;
10     }
11     
12     public void delete(T obj){
13         
14     }
15     
16     public void delete(int id){
17         
18     }
19     
20     public T update(T obj){
21         return null;
22     }
23     
24     public Set<T> findByConditions(String where){
25             return null;
26     }
27     
28 }

五、通过反射获取泛型的实际类型参数

这个问题从来没有思考过,用过框架,但是并没有仔细思考过,为什么,底层是怎么杨实现的呢,今天看了张老师的视频,感觉自己就是在工作上挣钱,并没有投入太多的热情在学习中,在工作中,这也许就是为什么会和别人的差距越来越大,这是很值得深思的问题啊,看一下具体的怎么实现的,记住这个,以后再用框架的时候,可以多思考一下

 1     // 为什么用Vector.class而不用Vector<Date>.class 后面的本身就是错误的,在编译之后会将泛型擦除掉,其实本质还是Vector
 2     Method applyMethod = GenericTest.class.getMethod("applyVector", Vector.class);
 3     Type[] types = applyMethod.getGenericParameterTypes();
 4     ParameterizedType pType = (ParameterizedType) types[0];
 5     System.out.println(pType.getRawType());
 6     System.out.println(pType.getActualTypeArguments()[0]);
 7 
 8     // 将需要获取泛型类型的变量交给一个方法,当做参数来获取其中参数的实际类型
 9     public static void applyVector(Vector<Date> v1){
10         
11     }

这里具体是在哪些框架中应用到了,现在还没有太清楚的,希望以后再看框架源码的时候,可以重新得到启发。

猜你喜欢

转载自www.cnblogs.com/ssh-html/p/10780398.html