这篇主要总结数组反射。先看一段代码:
public static void main(String[] args) throws Exception{
Object[] obj={1,"sss",'d'};
printObj(obj);
}
private static void printObj(Object obj) {
Class clazz = obj.getClass();
if(clazz.isArray()){
for(int i=0;i<Array.getLength(obj);i++)
System.out.println(Array.get(obj, i));
}else{
System.out.println(obj);
}
}
}
主函数中调用了printObj()方法,此方法的目的是输出传入的obj参数,如果是普通类型的参数就直接输出,如果是数组的话就输出数组的所以元素。首先通过getClass()获得参数的字节码,然后通过isArray()判断参数是否是数组,如果是数组的话则用一个for循环输出所有的数组元素。这里用到了一个类,java.lang.reflect.Array类,此类中有许多操作数组的方法,通过getLength()获得数组的长度,然后再通过get()方法取得数组的元素。
除此之外,JAVA5引入了可变参数的功能,比起JAVA4中需要传递数组类型的参数来确认是哪一个同名方法更加方便,但因为要向下兼容,所以如果把一个数组整体当作一个参数传递,编译器就会以JAVA4的方法来解析,即寻找有与数组长度相同的参数个数的同名函数与之匹配,结果是可能出现意想不到的错误。如:
public static void main(String[] args) throws Exception{
Test3 test = new Test3();
String[] string = {"1","2"};
Method method = test.getClass().getMethod("fun",String[].class);
method.invoke(test,string);
}
}
class Test3{
public void fun(String[] str){
System.out.println("this is a function!");
};
}
我用反射的方法调用了test的fun()方法,此方法接受的参数是一个字符串数组,所以我在invoke(test,string)第二个参数传入了一个字符串数组,但运行结果报错了。因为按照JAVA4的处理方法,invoke()方法里面的传入参数string被拆分成两个字符串,分别当成不同的传入参数,即需要匹配的Test3方法为fun(String str1,String str2),所以报错了。解决办法有两种:
1:
public static void main(String[] args) throws Exception{
Test3 test = new Test3();
String[] string = {"1","2"};
Method method = test.getClass().getMethod("fun",String[].class);
method.invoke(test,new Object[]{string});
}
}
class Test3{
public void fun(String[] str){
System.out.println("this is a function!");
};
}
把invoke方法中的string再套上一层数组,JAVA在拆开外层的Object数组后就能把string这个字符串数组当作一个参数传入了。
2:
public static void main(String[] args) throws Exception{
Test3 test = new Test3();
String[] string = {"1","2"};
Method method = test.getClass().getMethod("fun",String[].class);
method.invoke(test,(Object)string);
}
}
class Test3{
public void fun(String[] str){
System.out.println("this is a function!");
};
}
把string参数强制转换成Object类型,转换后便不是数组类型的传入参数,所以就会按照JAVA5的方法处理。