有什么方法可以调用私有方法吗?

问题

我有一个使用XML和反射的类将Objects返回到另一个类。

通常这些对象是外部对象的子字段,但有时它是我想要动态生成的东西。我尝试过类似的东西,但无济于事。我相信这是因为Java不允许你访问private方法进行反射。

Element node = outerNode.item(0);
String methodName = node.getAttribute("method");
String objectName = node.getAttribute("object");

if ("SomeObject".equals(objectName))
    object = someObject;
else
    object = this;

method = object.getClass().getMethod(methodName, (Class[]) null);

如果提供的方法是private,则它失败并带有aNoSuchMethodException。我可以通过制作方法public或者使另一个类从中派生它来解决它。

长话短说,我只是想知道是否有办法通过反射访问aprivate方法。


#1 热门回答(246 赞)

你可以使用反射调用私有方法。修改已发布代码的最后一位:

Method method = object.getClass().getDeclaredMethod(methodName);
method.setAccessible(true);
Object r = method.invoke(object);

有几点需要注意。首先,getDeclaredMethod将只查找current9038833中声明的方法,而不是从超类型继承。因此,如有必要,遍历具体的类层次结构。其次,aSecurityManager可以防止使用thesetAccessible方法。因此,它可能需要以aPrivilegedAction运行(使用AccessControllerSubject)。


#2 热门回答(33 赞)

使用getDeclaredMethod()获取私有Method对象,然后使用method.setAccessible()允许实际调用它。


#3 热门回答(22 赞)

如果方法接受非原始数据类型,则可以使用以下方法来调用任何类的私有方法:

public static Object genericInvokMethod(Object obj, String methodName,
            int paramCount, Object... params) {
        Method method;
        Object requiredObj = null;
        Object[] parameters = new Object[paramCount];
        Class<?>[] classArray = new Class<?>[paramCount];
        for (int i = 0; i < paramCount; i++) {
            parameters[i] = params[i];
            classArray[i] = params[i].getClass();
        }
        try {
            method = obj.getClass().getDeclaredMethod(methodName, classArray);
            method.setAccessible(true);
            requiredObj = method.invoke(obj, params);
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

        return requiredObj;
    }

接受的参数是obj,methodName,接受的参数计数和参数。例如

public class Test {
private String concatString(String a, String b) {
    return (a+b);
}
}

方法concatString可以作为调用

Test t = new Test();
    String str = (String) genericInvokMethod(t, "concatString", 2, "Hello", "Mr.x");