Java反射---方法的反射、深入了解泛型

来源:互联网 发布:阿里云2016年宕机问题 编辑:程序博客网 时间:2024/04/28 18:54

方法的反射

invoke(对象,参数列表)
1、获得类类型
Class data = a.getClass();

2、获得方法
- getMethod(“方法名”,可变参数(Class[])) 获得自身和继承的public方法
- getDeclaredMethod(“方法名”,可变参数(Class[])) 获得自身的所有方法,不包括继承的方法

    Method print = data.getMethod("print", new Class[]{int.class,int.class});    Method print2 = data.getDeclaredMethod("print", new Class[]{String.class,String.class});

3、执行方法
invoke(对象obj,参数列表) 使用对象obj调用方法

        print.invoke(a, 10, 20);        print2.invoke(a, new String[]{"hello","world"});

如果没有参数,可以不写:

        Method print3 = data.getMethod("print");        print3.invoke(a);

通过反射了解泛型的本质

所谓泛型,是在编译阶段判断变量类型是否满足泛型要求。例如ArrayList<Stirng> 在编译时如果add一个int类型的变量肯定会出错,但是,在编译完成后,不同类型的泛型实际上是一样的。如下:

        ArrayList list1 = new ArrayList();        ArrayList<String> list2 = new ArrayList<String>();        Class c1 = list1.getClass();        Class c2 = list2.getClass();        System.out.println(c1 == c2); // true

说明不同类型的泛型在编译后是一样的,泛型只是帮助判断变量类型的一种机制。
String类型的泛型,只能加入String类型的变量,所以集合中保存的变量原本都是String类型。而Object类型的泛型可以加入任意类型的变量,所有变量都会被转换为Object类型后保存,所以在去变量时,需要手动的将Object类型强制转换为指定的类型。

反射机制是在编译之后完成的。所以可以利用反绕过泛型的编译:

        Method add = c2.getMethod("add", Object.class);        add.invoke(list2, 100);        System.out.println(list2.size());       // 1        System.out.println(list2);              // [100]

上面的例子在String类型的泛型集合中加入了整形变量,说明泛型的类型检查是在运行之前进行的。编译过后不会有泛型的类型检查,所以不会报错。另外我在编写的过程中还发现一个细节:

        Method add = c2.getMethod("add", String.class);

上面这行代码会报错,c2是ArrayList<String>的类类型。一个String类型的泛型集合却无法获得String类型的add方法。原因是在编译过后所有类型会被擦除。在ArrayList的源码中用一个Object类型的数组来储存数据。编译过变量会被保存在Object数组中,所以add方法中的类型也会被转化为Object

    transient Object[] elementData; // non-private to simplify nested class access

个人理解,欢迎指正

参考资料:反射

阅读全文
0 0
原创粉丝点击