首页 > 学院 > 开发设计 > 正文

反射-------通过反射跳过泛型编译器运行报异常的问题答案

2019-11-14 15:41:05
字体:
来源:转载
供稿:网友

    之前在博问中有问过一个问题  一直没有解决 ,今天偶然想到一个问题并联想到之前然后找到了答案。下面是代码问题:

 1 public static void main(String[] args) throws IllegalaccessException, IllegalArgumentException, InvocationTargetException, NoSuchMethodException, SecurityException { 2         //定义一个String类型的list1的集合 3          4         List<String> list1=new ArrayList<String>(); 5         //定义一个Integer类型的list2的集合 6          7         List<Integer> list2=new ArrayList<Integer>(); 8         System.out.PRintln(list1.getClass()==list2.getClass()); 9         System.out.println(list1.getClass());10         System.out.println(list2.getClass());11         12         list2.getClass().getMethod("add",Object.class).invoke(list2, "abc");13         list2.add(2);14         15         list1.add("ab");16         list1.getClass().getMethod("add", Object.class).invoke(list1, 1);        17         18         for(Iterator<Integer> it=list2.iterator();it.hasNext();){19             //Object i=it.next();20             System.out.print(it.next());21         }22         System.out.println();23         /*24         for (Object s : list1) {25             System.out.print(s+" ");26         }*/27         //????类型转换异常    28         for(Iterator<String> it=list1.iterator();it.hasNext();){29             //Object i=it.next();30             System.out.print(it.next());31         }32     }
如上面的代码 运行的话 会出现 往一个Integer类型的集合中添加一个String类型的元素 通过反射 可以跳过泛型检查的编译器 并且可以 成功遍历
而相反 往一个String类型的集合中 通过反射 插入一个Integer的数据 在遍历的时候会出现类型转换的异常
trueclass java.util.ArrayListclass java.util.ArrayListabc2abException in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String    at cn.itcast.Genericity.ReflectThroughtGenericity.main(ReflectThroughtGenericity.java:52)

 

原因:

如果遍历时候将其用Object的类型来遍历时不会出错的,最终的原因是在输出语句System.out.print()中 查看源代码会发现:
 public void print(boolean b) {        write(b ? "true" : "false");    }    public void print(char c) {        write(String.valueOf(c));    }    public void print(int i) {        write(String.valueOf(i));    }      public void print(long l) {        write(String.valueOf(l));    }    public void print(float f) {        write(String.valueOf(f));    }    public void print(String s) {        if (s == null) {            s = "null";        }        write(s);    }    public void print(Object obj) {        write(String.valueOf(obj));    }

    在print的重载方法中 有String类型为参数的方法 而没有Integer类型为参数的方法 ,因此会在遍历时候 若是遍历Integer的集合 他会自动向上提升到Object也就没有了类型转换异常的问题,而String的集合遍历时会调用String参数类型的方法 当遍历到Integer的数据时则会出现类型转换异常 。



 


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表