如何使用反射来调用带有原始参数的方法?

时间:2018-12-24 00:47:06

标签: java reflection

我正在尝试创建方法invokeMethod(String methodName, Object...args),该方法从当前实例的超类中调用一个方法。我尝试了以下实现。

    public void invokeMethod(String methodName, Object...args) {
        //Array to hold the classes of the arguments
        Class<?>[] classes = new Class<?>[args.length]; 
        //Initialize each class in the array to the class of each argument 
        for(int i = 0; i < args.length; i++)
            classes[i] = args[i].getClass();
        try {
            //find the method
            Method m = this.getClass().getMethod(methodName, classes);
            //invoke the method
            m.invoke(this, args);
        } catch (NoSuchMethodException | SecurityException | IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
            e.printStackTrace();
        }
    }

此实现的问题是,如果我尝试调用具有原始参数的方法,则会得到NoSuchMethodException,因为它正在寻找一种类型与包装类等效的参数的方法。

例如,如果我尝试通过尝试line(float, float, float, float)来调用带有签名invokeMethod("line", 50f, 50f, 50f, 50f)的方法,则会得到类似于

的异常。
java.lang.NoSuchMethodException: DynamicSketch.line(java.lang.Float, java.lang.Float, java.lang.Float, java.lang.Float)
at java.base/java.lang.Class.getMethod(Class.java:2109)
at DynamicSketch.invokeMethod(DynamicSketch.java:32)
at DynamicSketch.setup(DynamicSketch.java:19)
at processing.core.PApplet.handleDraw(PApplet.java:2412)
at processing.awt.PSurfaceAWT$12.callDraw(PSurfaceAWT.java:1557)
at processing.core.PSurfaceNone$AnimationThread.run(PSurfaceNone.java:316)

反正我的invokeMethod方法可以与原始参数一起使用吗?

编辑:解决方案here不起作用,因为执行我的方法时我不完全知道签名中的原始类型。我希望能够使用我的方法执行诸如size(int, int)line(float, float ,float ,float)之类的方法,并且链接中的解决方案对此不起作用。我看到的唯一解决方案是为超类中的每个可能方法定义一个if语句,然后在链接中使用该解决方案,但我希望使用一种较简单的方法。

1 个答案:

答案 0 :(得分:4)

将诸如1010.0f之类的参数传递到方法中时,由于参数类型为Object...,它们会自动换行。

因此,您需要检查这些包装器类型并拆开它们。您的for循环如下所示:

for(int i = 0; i < args.length; i++) {
    if (args[i].getClass() == Integer.class) {
        classes[i] = int.class;
    } else if (args[i].getClass() == Float.class) {
        classes[i] = float.class;
    } else if (args[i].getClass() == Double.class) {
        classes[i] = double.class;
    } else {
        classes[i] = args[i].getClass();
    }
}

我在这里只添加了3个案例,您可以自己添加其他5个案例。

这意味着您现在不能使用包装器类型参数调用方法。如果您也想拨打这些电话,则需要

  1. 解开所有包装类型
  2. 尝试查找具有那些非包装类型的方法
  3. 如果找不到,请重新包装
  4. 使用包装器类型找到a方法。

编辑:

正如Holger在评论中建议的那样,您还可以通过使用

让JVM为您找到合适的方法。
new Statement(this, methodName, args).execute();

Docs for the Statement class.