原文链接:http://tutorials.jenkov.com/java-reflection/private-fields-and-methods.html
尽管大家普遍认为通过 Java 反射是不能访问到私有的字段和方法的,事实上这是可以的,其实也并不难。。其实通过 Java 反射访问私有的字段和方法非常便于单元测试。本文将会向大家介绍怎样通过 Java 反射来获取私有的字段和方法。
注意:本文讲的方法只适用于孤立的 Java 应用,比如 unit test 和 regular applications 。如果我们想要在 Java Applet 程序上使用,我们得篡改 Java 的 SecurityManager 机制。但是,我们很少需要这种情况,这种情况已经远远的超出了本文所讨论的范围。
Accessing Private Fields
public class PrivateObject { private String privateString = null; public PrivateObject(String privateString) { this.privateString = privateString; } }
PrivateObject privateObject = new PrivateObject("The Private Value"); Field privateStringField = PrivateObject.class. getDeclaredField("privateString"); privateStringField.setAccessible(true); String fieldValue = (String) privateStringField.get(privateObject); System.out.println("fieldValue = " + fieldValue);
这段代码示例会输出 "fieldValue=The Private Value" ,这个字符串正是在开头的代码示例中 创建的 PrivateObject 的实例 privateObject 的私有字段的值。
注意: PrivateObject.class.getDeclaredField("privateString") 这个方法的使用,调用这个方法会返回那个私有字段。这个方法只会返回在这个类中声明的字段,而不会返回它的父类或者是它实现的接口中的字段。
也注意粗体标注的那一行代码,在Java 反射中,我们通过调用 Field.setAccessible(true) 关闭了 Field 的这个实例的访问范围。现在我们可以访问任意一个字段了,不管它是 Private , protected ,package scope ,甚至是这些范围之外的。但我们仍然不能通过正常的代码(比如通过 privateObject.privateString 这样来访问 privateString 这个私有字段)来访问字段,编译器不允许我们这样做。
Accessing Private Methods
为了访问私有的方法我们需要通过调用 Class.getDeclaredMethod(String name) 或者是 Class.getDeclaredMethods() 。Class.getMethod(String name) 和 Class.getMethods() 只能获取到 public 方法,所以我们不可能通过它们来获取到私有的方法。下面是一个简单的例子,在这个简单的例子当中用了一个简单的类,这个简单的类只有一个私有字段和一个有String类型的构造方法和一个私有的方法,我们将会通过 Java 反射来访问 getPrivateString() 这个私有方法。
public class PrivateObject { private String privateString = null; public PrivateObject(String privateString) { this.privateString = privateString; } private String getPrivateString(){ return this.privateString; } }
PrivateObject privateObject = new PrivateObject("The Private Value"); Method privateStringMethod = PrivateObject.class. getDeclaredMethod("getPrivateString", null); privateStringMethod.setAccessible(true); String returnValue = (String) privateStringMethod.invoke(privateObject, null); System.out.println("returnValue = " + returnValue);
这段代码示例会输出 "returnValue=The Private Value" ,这个字符串正是通过调用在开头的代码示例中 创建的 PrivateObject 的实例 privateObject 的私有方法(getprivateString) 而返回的的值。
注意 PrivateObject.class.getDeclaredMethod("privateString") 这个方法的使用,调用这个方法会返回那个私有的字段。这个方法只会返回在这个类中声明的方法,而不会返回它的父类或者是它实现的接口中的方法。
也注意粗体标注的那一行代码,在Java 反射中,我们通过调用 Method.setAccessible(true) 关闭了 Method 的这个实例的访问范围。现在我们可以访问任意一个方法了,不管它是 Private , protected ,package scope ,甚至是这些范围之外的。但我们仍然不能通过正常的代码(比如通过 getPrivateString(); 这样来访问 getPrivateString 这个私有方法)来访问它,编译器不允许我们这样做。