黑马程序员技术交流社区

标题: 一个关于反射、泛型、集合的问题 [打印本页]

作者: winsking    时间: 2013-10-10 11:41
标题: 一个关于反射、泛型、集合的问题
本帖最后由 winsking 于 2013-10-11 10:24 编辑

昨天有个哥们提问了一个反射问题,http://bbs.itheima.com/forum.php?mod=viewthread&tid=87739,当时在网吧,不能实践操作一下,就想当然的回复了,回家后在自己电脑上操作后发现很是不解,现在抽空跑网吧特地发帖来求解答:先发代码:
  1. import java.lang.reflect.InvocationTargetException;
  2. import java.lang.reflect.Method;
  3. import java.util.ArrayList;
  4. import java.util.List;

  5. public class Sss {

  6.         public static void main(String[] args) throws Exception
  7.         {
  8.                 ArrayList<Integer> s2 = new ArrayList<Integer>();               
  9.                 Method m2 = s2.getClass().getMethod("add", Object.class);
  10.                 m2.invoke(s2, "abc");
  11.                 System.out.println(s2.get(0));//打印出 abc
  12.                
  13.                 ArrayList<Person> s3 = new ArrayList<Person>();               
  14.                 Method m3 = s3.getClass().getMethod("add", Object.class);
  15.                 m3.invoke(s3, 7);
  16.                 System.out.println(s3.get(0)); //打印出7
  17.                
  18.                 ArrayList<String> s = new ArrayList<String>();               
  19.                 Method m = s.getClass().getMethod("add", Object.class);
  20.                 m.invoke(s, 1);
  21.                 System.out.println(s.get(0)); //报错ClassCastException
  22.         }
  23. }
  24. class Person{
  25.         private String name;
  26.         Person(String name){
  27.                 this.name = name;
  28.         }
  29. }
复制代码
发现当泛型设置成String时,m.invoke(s, 1);自动装箱貌似没问题,可以存进集合,可以直接打印集合,但是下一句:System.out.println(s.get(0)); //报错ClassCastException,自定义一个类Person当做泛型也是没问题,就是<String>时,调用get()方法出错,想不通,去找Java源码看了,看不出什么东西,自己功力尚浅吧  呵呵
下面是文档中 ClassCastException的解释:

public class ClassCastExceptionextends RuntimeException当试图将对象强制转换为不是实例的子类时,抛出该异常。例如,以下代码将生成一个 ClassCastException:
     Object x = new Integer(0);
     System.out.println((String)x);

求解答{:soso_e100:}

看来飘落同学的说法很对,泛型设置为基本类型包装类或者一个其他类时,打印语句调用的参数都是Object x,只有泛型设置为String时是String x,所以这时候特殊,这时把23行改成  System.out.println((Object)s.get(0));就不会报错了

作者: 占琳    时间: 2013-10-10 12:15
经过实测了   m.invoke(s, 1); 你给的是String类型,但是1是int类型 ,所以出错  加上引号即可
作者: 飘落    时间: 2013-10-10 14:32
抛异常是方法重载惹的祸,
在编译时,根据集合上的泛型<String>,println()选择的是
public void println(String x)
而不是public void println(Object x)。
这样运行时接收的是Integer,而需要的是String,
那可不是要抛类型转换异常。




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2