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

She*_*oss 137 java reflection private

我有一个使用XML和反射的Object类将s 返回到另一个类.

通常这些对象是外部对象的子字段,但偶尔它是我想要动态生成的东西.我尝试过类似的东西,但无济于事.我相信这是因为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);
Run Code Online (Sandbox Code Playgroud)

如果提供的方法是private,它失败了NoSuchMethodException.我可以通过制作方法public或使另一个类从中派生它来解决它.

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

eri*_*son 282

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

Method method = object.getClass().getDeclaredMethod(methodName);
method.setAccessible(true);
Object r = method.invoke(object);
Run Code Online (Sandbox Code Playgroud)

有几点需要注意.首先,getDeclaredMethod只会查找当前声明的方法Class,而不是从超类型继承.因此,如有必要,遍历具体的类层次结构.其次,a SecurityManager可以防止使用该setAccessible方法.因此,它可能需要作为PrivilegedAction(使用AccessControllerSubject)运行.

  • 不,当您设置辅助功能时,它仅适用于该实例.只要你不让那个特定的Method对象从你的控件中逃脱,它就是安全的. (15认同)
  • 那么,如果可以从课外调用私有方法,那么有什么意义呢? (6认同)
  • @PeterAjtai 抱歉回复晚了,但可以这样想:现在大多数人都会锁门,即使他们知道锁可能会被轻易打破或完全绕过。为什么?因为它有助于让大多数诚实的人保持诚实。您可以将“私有”访问视为类似的角色。 (5认同)
  • 当我在过去做过这个时,我在调用方法后也调用了method.setAccessible(false),但我不知道这是否有必要. (2认同)
  • 还要确保调用`getDeclaredMethod()`而不是`getMethod()` - 这对私有方法不起作用. (2认同)

Mih*_*der 33

使用getDeclaredMethod()获得的私有方法的对象,然后使用method.setAccessible()允许实际调用它.

  • 接受的答案实际上已经完成. (7认同)

gKa*_*aur 24

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

public static Object genericInvokeMethod(Object obj, String methodName,
            Object... params) {
        int paramCount = params.length;
        Method method;
        Object requiredObj = null;
        Class<?>[] classArray = new Class<?>[paramCount];
        for (int i = 0; i < paramCount; 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;
    }
Run Code Online (Sandbox Code Playgroud)

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

public class Test {
private String concatString(String a, String b) {
    return (a+b);
}
}
Run Code Online (Sandbox Code Playgroud)

方法concatString可以作为调用

Test t = new Test();
    String str = (String) genericInvokeMethod(t, "concatString", "Hello", "Mr.x");
Run Code Online (Sandbox Code Playgroud)

  • 为什么需要_paramCount_?你不能只使用*params.length*? (7认同)

Kad*_*LAB 8

你可以使用 Spring 的 ReflectionTestUtils ( org.springframework.test.util.ReflectionTestUtils )来做到这一点

ReflectionTestUtils.invokeMethod(instantiatedObject,"methodName",argument);
Run Code Online (Sandbox Code Playgroud)

示例:如果您有一个带有私有方法的类 square(int x)

Calculator calculator = new Calculator();
ReflectionTestUtils.invokeMethod(calculator,"square",10);
Run Code Online (Sandbox Code Playgroud)