A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 水云间 中级黑马   /  2013-5-29 13:27  /  1579 人查看  /  1 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 水云间 于 2013-5-29 13:28 编辑

一.java泛型的引入和作用
在JDK1.5以后,引入了泛型的概念,泛型的出现可以有效的解决程序安全机制问题,将程序运行时的问题转到编译时期,使程序员在编译时期就能将可能出现的问题解决,这样提高了程序代码的健壮性,同时引入泛型以后,可以避免令程序员头痛的强制类型转换问题,是程序编译起来更简洁,下面我们先看看在泛型出现以前,程序员可能碰到的问题,如下代码所示:
  1. package cn.itcast.generic;

  2. import java.util.ArrayList;
  3. import java.util.List;

  4. public class GenericTest {

  5.         /**
  6.          * @param args
  7.          */
  8.         public static void main(String[] args) {
  9.                 //在泛型出现以前,一个集合类可以同时存有多种类型的数据,在取数据时不仅需要强制类型
  10.                 //转换,还会出现安全机制
  11.                 List list = new ArrayList();
  12.         list.add(1);
  13.         list.add("abc");
  14.         list.add(1>2);
  15.         //因为list集合类可以装各种类型数据,返回的是Object,因此需要类型的强制转换
  16.         int value = (Integer) list.get(0);
  17.         System.out.println(value);
  18.    
  19.       ArrayList<String>();
  20.         list1.add("123");
  21.      
  22.      
  23.         }

  24. }
复制代码
对上述代码进行分析,由于ArrayList可以存储各种数据类型,如Integer,String,StringBuffer等,因此,程序员可以往里面存储各种数据,编译器在编译时不会报错,可是当需求上只能往里面加入一种类型(如String类型)的数据,而程序员却往里面存储了一个非String类型的数据,可是程序员在编译时编译器并没有报错,当交给用户运行时,却出现了错误,将给程序的修改带来,而加入泛型以后,编译器会自动扫描语法,当出现不是需要的类型时,编译器就会自动提醒程序员,因此,泛型的引入极大的提高了代码的安全性,同时,使用了泛型还能让程序员不必进行强制类型转换,减少了程序员的工作量,下面用泛型来改写上述代码:
  1. package cn.itcast.generic;

  2. import java.util.ArrayList;
  3. import java.util.List;

  4. public class GenericTest {

  5.         /**
  6.          * @param args
  7.          */
  8.         public static void main(String[] args) {
  9.         //泛型出现以后,在定义一个集合的时候,就为其指定了存储类型,当你往集合类存储的数据
  10.         //不是指定的类型时,编译器会报错,这样就把运行时的问题转到编译时期,提高程序的安全性
  11.         List<String> list1 = new ArrayList<String>();
  12.         list1.add("123");
  13.         list1.add("abc");
  14.         //由于在集合中已经指定了存储类型,因此这里不需要强制类型转换
  15.         String str = list1.get(1);
  16.         System.out.println(str);
  17.         }
  18. }
复制代码
二.泛型内部的运行原理和深层次的应用
先看一组代码:
  1. package cn.itcast.generic;

  2. import java.lang.reflect.InvocationTargetException;
  3. import java.util.ArrayList;
  4. import java.util.List;

  5. public class Theory {

  6.         /**
  7.          * @param args
  8.          *  
  9.          */
  10.         public static void main(String[] args) throws Exception {
  11.                 //建立一个存储字符串类型的集合
  12.                 List<String> list1 = new ArrayList<String>();
  13.         
  14.         //建立一个存储Integer类型的集合
  15.         List<Integer> list2 = new ArrayList<Integer>();
  16.         
  17.         //比较list1和list2的字节码是否相同
  18.         System.out.println(list1.getClass()==list2.getClass());
  19. }
  20. }
复制代码
编译,运行文件,我们可以在控制台上看到打印一个true,从而验证了尽管在定义时,两个集合中所存储的类型不一致,但是当编译完成后生成的Class类的对象却是一致的,其实他内部的工作原理是,编译器在编译时会检查他的存储类型,从而检查语法有没有问题,而当编译通过后,他会自动去除参数类型,因此他们生成的Class对象才会一致,并且和原型一致,由此,我们便可以用反射的方式来操作一个集合,使得该集合能存储和本集合参数类型不一致的数据,如下面代码所示:
  1. package cn.itcast.generic;

  2. import java.lang.reflect.InvocationTargetException;
  3. import java.util.ArrayList;
  4. import java.util.List;

  5. public class Theory {

  6.         /**
  7.          * @param args
  8.          *  
  9.          */
  10.         public static void main(String[] args) throws Exception {
  11. List<Integer> list2 = new ArrayList<Integer>();
  12. //通过反射可以用list2来存储字符串类型的数据
  13.         
  14.         list2.getClass().getMethod("add", Object.class).invoke(list2, "abc");
  15.         
  16.         //打印集合中的数据
  17.         System.out.println(list2.get(0));
  18.         

  19.         }

  20. }
复制代码
三.java泛型通配符和泛型限定
泛型通配符用“?”表示,它代表着该类或方法可以接收任意类型的变量,下面来举个例子来说明通配符的概念,如下所示:
  1. package cn.itcast.generic;

  2. import java.util.ArrayList;
  3. import java.util.List;

  4. public class Generic {

  5.         /**
  6.          * ?通配符可以引用其他各种类型化的参数,?通配符定义的变量主要用作引用,可以调用与参数
  7.          * 无关的方法,不能调用与参数有关的方法
  8.          * @param args
  9.          */
  10.         public static void main(String[] args) {
  11.                 //建立一个存储String类型的集合
  12.                 List<String> list1 = new ArrayList<String>();
  13.                 //添加两个元素
  14.                 list1.add("abc");
  15.                 list1.add("xyz");
  16.                 //建立一个存储Integer类型的集合
  17.                 List<Integer> list2 = new ArrayList<Integer>();
  18.                 //添加两个元素
  19.                 list2.add(12);
  20.                 list2.add(13);
  21.                
  22.                 //打印list集合类的元素
  23.                 printLisst(list1);
  24.                 //打印list集合类的元素
  25.                 printLisst(list2);
  26.         }

  27.         private static void printLisst(List<?> list) {
  28.                 System.out.println(list.size());
  29.                 for(Object obj :list){
  30.                         System.out.println(obj);
  31.                 }
  32.                
  33.         }

  34. }
复制代码
如上代码所示,我们定义了两个ArrayList集合,分别用来存储String类型的数据和Integer类型的数据,并且分别往这两个集合内存入两个元素,而后有定义了一个方法,该方法接收一个List集合,而参数类型为<?>,显然该类集合可以引用各种类型的参数,而该方法用来打印集合的长度,和迭代集合的元素,这个例子很好的说明了通配符的作用。下面是对通配符的一个总结:
通配符可以引用其他各种类型化的参数,?通配符定义的变量主要用作引用,可以调用与参数 无关的方法,不能调用与参数有关的方法 。
四.泛型的一个应用案例

学完泛型后,用泛型来做一个小案例,即泛型和HashMap的结合,如何去HashMap的键值对代码如下:
  1. package cn.itcast.generic;

  2. import java.util.HashMap;
  3. import java.util.Map;
  4. import java.util.Set;
  5. import java.util.Map.Entry;

  6. public class GernericApplication {

  7.         /**
  8.          * 利用泛型来取出操作HashMap集合
  9.          * @param args
  10.          */
  11.         public static void main(String[] args) {
  12.                 //定义一个HashMap
  13.                 HashMap<String,Integer> maps = new HashMap<String,Integer>();
  14.                 //添加三条数据
  15.                 maps.put("prt", 23);
  16.                 maps.put("wb",20);
  17.                 maps.put("wjj", 25);
  18.                 //调用entrySet方法获得集合映射关系
  19.                 Set<Map.Entry<String, Integer>> entrySet = maps.entrySet();
  20.                 //迭代去数据
  21.                 for(Entry<String,Integer> entry:entrySet){
  22.                         String key =  entry.getKey();
  23.                         int value = entry.getValue();
  24.                         System.out.println(key+"="+value);
  25.                 }
  26.         }

  27. }
复制代码

点评

很给力,赞一个  发表于 2013-6-1 00:33

评分

参与人数 1技术分 +1 收起 理由
Sword + 1

查看全部评分

1 个回复

倒序浏览
您需要登录后才可以回帖 登录 | 加入黑马