JAVA反射参数传递

时间:2021-05-07 16:27:03

引用:http://fish2700.blog.163.com/blog/static/130713192009103035723281/

使用Method反射调用函数时,我们通常会遇到以下几种情况:

 public void test(){

System.out.println("函数参数:0");

}

public void test(String str){

System.out.println("函数参数:1----------" + str);

}

public void test(String str1, String str2){

System.out.println("函数参数:2----------" + str1 +"   " + str2);

}

public void test(Object...objs){

System.out.print("函数参数:" + objs.length + "----------------");

for(Object o : objs ){

System.out.print(o.toString() + "    ");

}

}

而当我们使用Class.getMethod,则需要传递所调用函数的参数类型。查看Class.getMethod 的API可知,需要传递的类型被表示为一个可变参数。

我们知道,传递可变参数时,非序列参数会被编译成编列,即变成一个Object[]类型的数组,但是本身为序列的则会直接被转型Object[]数组。

那么,前三种情况按照要求传递,则传递给getMethod的参数会被转变为一个一维的参数列表的Object数组。第四种情况,其函数本身便要求传递一个可变参数,即一个Object[]类型的参数。如果我们按照正常方法传递,则此Object[]类型的参数会被直接转型使用,而我们最终传给函数的应该是一个二维的Ojbect数组,即Object[][]类型。getMethod方法的匹配过程是指寻找参数长度与Object数组的长度相等,且每个参数类型与Object数组每个数组项相同的方法。

所以,再这种情况下,我们应当对每四种情况下将要传递的参数进行一次包装,将其包装成一个二维的Object数组。方法如下:

Object[] obj = new Object[1];

    String[] strs = new String[]{"xiao","she", "qing"};

obj[0] = strs ;

此时的obj则是我们将要传给Class.getMethod的参数,而strs则是我们要传递给调用函数test(Object...objs)的参数。这里的obj长度为1是因为可变参数在没有参数传递之前的检查时的长度为1,被视为一元参数。

由于Spring使用的是Java代理,所以,在Spring中会经常遇到类似的问题。

具体代码如下:

package test;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class MethodDemo {

public void test(){
  System.out.println("函数参数:0");
 }
 
 public void test(String str){
  System.out.println("函数参数:1----------" + str);
 }
 
 public void test(String str1, String str2){
  System.out.println("函数参数:2----------" + str1 + " " + str2);
 }
 
 public void test(Object...objs){
  System.out.print("函数参数:" + objs.length + "----------------");
  for(Object o : objs ){
   System.out.print(o.toString() + " ");
  }
 }
 /**
  * @param args
  */
 /**
  * @param args
  */
 public static void main(String[] args) {
  
  //testMethod();
  printMethodType();
 }
 
 public static void printMethodType(){
  Method[] methods = MethodDemo.class.getMethods();
  Class[] cs;
  for(Method m : methods){
   System.out.println("----------------" + m.getName() + "----------------");
   cs = m.getParameterTypes();
   System.out.println(cs.length);
   for(Class c : cs){
    System.out.println(c.toString());
   }
  }
 }
 
 public static void testMethod(){
  MethodDemo demo = new MethodDemo();
  Method method;
  try {
   method = MethodDemo.class.getMethod("test", null);
   method.invoke(demo, null);
   System.out.println("--------------------------------------");

String content = "xiao";
   method = MethodDemo.class.getMethod("test", String.class);
   method.invoke(demo, content);
   System.out.println("--------------------------------------");

String str1 = "xiao";
   String str2 = "qing";
   method = MethodDemo.class.getMethod("test", String.class, String.class);
   method.invoke(demo, str1, str2);
   System.out.println("--------------------------------------");

Object[] obj = new Object[1];
   obj[0] = new String[]{"xiao", "she", "qing"};
   //obj[1] = new String[]{"xiao", "qing"};
   method = MethodDemo.class.getMethod("test", Object[].class);
   method.invoke(demo, obj);

} catch (SecurityException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  } catch (NoSuchMethodException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  } catch (IllegalArgumentException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  } catch (IllegalAccessException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  } catch (InvocationTargetException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }
 }

}