问:下面程序段注释部分运行结果分别是什么?为什么?
public class Test {
public void age(int age) {
System.out.println("int age="+age);
}
public void age(Integer age) {
System.out.println("Integer age="+age);
}
public static void main(String[] args) throws Exception {
Test obj = new Test();
Method m1 = obj.getClass().getMethod("age", int.class);
m1.invoke(obj, new Integer(27)); //1
m1.invoke(obj, 28); //2
Method m2 = obj.getClass().getMethod("age", Integer.class);
m2.invoke(obj, new Integer(27)); //3
m2.invoke(obj, 28); //4
Method m3 = obj.getClass().getMethod("age", Integer.TYPE);
m3.invoke(obj, new Integer(27)); //5
m3.invoke(obj, 28); //6
}
}
答:上面程序的运行结果如下。
int age=27
int age=28
Integer age=27
Integer age=28
int age=27
int age=28
因为在反射中基本类型的 class 和其对应包装类的 class 是不同的,所以在获得 Method 指定参数时需要精确指定参数的类型,即基本类型参数 int 是无法使用包装类型 Integer 来获得的,在 Java 反射中一定要注意这个问题。
这个问题中想反射调用基本类型参数方法除了可以通过 基本类型名.class
外还可以通过 对应包装类的静态字段 TYPE
获得,即通过调用 getMethod("age", Integer.TYPE).invoke(27);
来实现反射 int 类型的 age 方法,因为 int.class
等价于 Integer.TYPE
。
该问题的本质其实是因为 Java 中一切皆对象,而基本数据类型没有类的全限定名且没有 getClass 方法可供使用,所以如果想用 Class 类来表示基本数据类型的 Class 实例就得使用 Java 提供的内置实例,即:
Class clzss = byte.class;
Class clzss = short.class;
Class clzss = int.class;
Class clzss = long.class;
Class clzss = char.class;
Class clzss = float.class;
Class clzss = double.class;
Class clzss = boolean.class;
Class clzss = void.class;
此外 Java 的基本数据类型的包装类中都有一个名为 TYPE 的常量来表示对应包装类型的基本数据类型的 Class 实例,即:
Byte.TYPE = byte.class;
Short.TYPE = short.class;
Integer.TYPE = int.class;
Long.TYPE = long.class;
Char.TYPE = char.class;
Float.TYPE = float.class;
Double.TYPE = double.class;
Boolean.TYPE = boolean.class;
Void.TYPE = void.class;
结论:包装类.TYPE = 基本数据类型.class
同理对于数组类型的 Class 类型实例也是类似规则,譬如:
Class clzss = int[].class; //int[] 数组
Class clzss = String[].class; //String[] 数组
Class clzss = String[][].class; //String[][] 数组的数组