博客已迁移到CSDN《https://blog.csdn.net/qq_33375499》
一个类的主要成员时方法,辣么我们通过反射获取到一个类的所有方法信息后,总的寻找一种方式去操作调用这些方法,这样反射才有意义有意思。
Method对象有一个方法invoke。 public Object invoke(Object obj, Object... args) throws IllegalAccessException,IllegalArgumentException,InvocationTargetException。 该方法的解释API解释为:
对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。个别参数被自动解包,以便与基本形参相匹配,基本参数和引用参数都随需服从方法调用转换。 如果底层方法是静态的,那么可以忽略指定的 obj 参数。该参数可以为 null。 如果底层方法所需的形参数为 0,则所提供的 args 数组长度可以为 0 或 null。 如果底层方法是实例方法,则使用动态方法查找来调用它,这一点记录在 Java Language Specification, Second Edition 的第 15.12.4.4 节中;在发生基于目标对象的运行时类型的重写时更应该这样做。 如果底层方法是静态的,并且尚未初始化声明此方法的类,则会将其初始化。 如果方法正常完成,则将该方法返回的值返回给调用者;如果该值为基本类型,则首先适当地将其包装在对象中。但是,如果该值的类型为一组基本类型,则数组元素不 被包装在对象中;换句话说,将返回基本类型的数组。如果底层方法返回类型为 void,则该调用返回 null。
下面我用ArrayList 类来做了一个测试:
import java.lang.annotation.Annotation; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; import java.lang.reflect.Parameter; import java.util.ArrayList; /** * java反射 工具类 */ public class ClassUtil { /** * 通过反射操作对象方法 */ public static void method() { try { // 本例中传入的是ArrayList 对象,就以ArrayList 对象的add方法为例 ArrayList arrayList = new ArrayList(); // 1.获取该类的类类型 Class c = arrayList.getClass(); // 2.获取该类自己定义的所有方法 Method[] methods = c.getDeclaredMethods(); // 3.得到所有方法列表信息 for (Method method : methods) { System.out.print(method.getName() + "("); Class[] parameterTypes = method.getParameterTypes(); for (int i = 0, len = parameterTypes.length; i < len; i++) { System.out.print(parameterTypes[i].getName()); if (i != len - 1) { System.out.print(","); } } System.out.print(")\n"); } // 4.使用方法 invoke 来操作方法 add Method method = c.getMethod("add", new Class[]{Object.class}); method.invoke(arrayList, "测试1"); method.invoke(arrayList, 100); method.invoke(arrayList, 100.0f); // 5.测试打印 arrayList 对象 System.out.println("================" + arrayList); } catch (Exception e) { e.printStackTrace(); } } public static void main(String[] args) { ClassUtil.method(); } }
对于java集合中的泛型,是用于在编译阶段程序员输入验证的,我们可以通过反射的方式,来绕过这种输入验证。如上面的栗子,我们将 ArrayList<String> arrayList = new ArrayList(); 指定泛型为String类型,其他代码不变,你会发现,打印输出的结果还是一样。