使用反射调用带有数组参数的方法

16

我正在尝试编写一个方法,该方法通过将字符串数组作为参数传递给方法,执行另一个类中的静态方法。

以下是我的代码:

public static void
executeStaticCommand(final String[] command, Class<?> provider)
{
    Method[] validMethods = provider.getMethods();

    String javaCommand = TextFormat.toCamelCase(command[0]);

    for (Method method : validMethods) {
        if (method.getName().equals(javaCommand)) {
            try {
                method.invoke(null, new Object[] { new Object[] { command } });
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            } catch (InvocationTargetException e) {
                Throwable ex = e.getCause();
                ex.printStackTrace();
            }
            break;
        }
    }
}

这样做:

String[] args = new String[] { "methodName", "arg1", "arg2" }; 
executeStaticCommand(args, ClassName.class);

应该执行这个:

public class ClassName {
    public static void methodName(String[] args) {
        assert args[1].equals("arg1");
    }
}

然而我遇到了 IllegalArgumentException 异常。

3个回答

25

您有两个问题:

  1. 目标参数类型为String[],但是您正在传递Object[]
  2. 您正在将整个命令数组作为参数传入,其中包括方法名称

这些问题都在内部的try块中,因此我只显示该代码。

String[] args = Arrays.copyOfRange(command, 1, command.length - 1);
method.invoke(null, new Object[]{args}); // must prevent expansion into varargs

感谢Perception提醒我关于可变参数的问题。


2
刚刚在打这个代码,还有一个额外的改进可以做。你需要将调用方法更改为 method.invoke(null, new Object[] {args}) 才能使其正常工作。 - Perception
@Perception 但是目标方法参数是String[],而不是String, String。这样行吗(我这里没有Java编辑器)? - Bohemian
2
它不起作用是因为数组被扩展到可变参数中,导致签名和参数类型不匹配。 - Perception
第二点是故意的,而且 method.invoke(null, new Object[]{ command }); 是我最初的写法(在另一个 new Object[]{} 中包装之前)。只有当 String[] 数组中只有一个元素时才有效。 - azz
你能试试Pshemo的建议,使用(Object)args吗?同时,将方法名作为参数传递给方法似乎不是一个好主意。为什么方法需要传递自己的名称呢? - Bohemian
没关系...事实证明问题出在我从STDIN构建字符串数组的方式上。我不小心包含了每个参数结尾和下一个参数之间的空格。这样做其实是可以工作的,但问题出在别处... - azz

1
你尝试调用的方法期望一个字符串数组,但是你正在传递一个对象数组作为参数。请将其更改为字符串数组,或者如果该方法期望对象,则可以传递任何类型。
method.invoke(null,(Object)command);

当我使用(Object)command时,如果command.length为1,则方法会被正确调用,但如果它> 1,则不会调用任何方法。 - azz

0
基于这个问题, 看起来应该调用这个函数。
 method.invoke(null, command);

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接