咖啡伴侣

呆在上海
posts - 163, comments - 156, trackbacks - 0, articles - 2

通过反射修改类的私有字段值,调用私有方法

Posted on 2008-06-21 11:40 oathleo 阅读(359) 评论(0)  编辑  收藏 所属分类: Java

下面这个例子在实际项目中一般都不会这么用,只是用来说明怎么通过反射修改类的私有字段的值.

有一个类TestData:

public class TestData {
 private String name = "1";
 
 public String getName() {
  return name;
 }
}
在运行的时候怎么来修改name的值呢?

public class TestReflection extends TestCase {
 public void testSetPrivateField() throws Exception {
  TestData data = new TestData();
  System.out.println(data.getName());
  Assert.assertEquals("1", data.getName());
  Field f = data.getClass().getDeclaredField("name");
  f.setAccessible(true);
  f.set(data, "2");
  System.out.println(data.getName());
  Assert.assertEquals("2", data.getName());
 }
}
运行结果:
1
2
其中,最关键的代码是:
f.setAccessible(true);
这行代码把对象data上的name字段设置为public访问属性.

既然私有字段可以这样访问,那么,类似的,私有方法也可以这样调用!
改一下TestData:

public class TestData {
 private String name = "1";
 
 public String getName() {
  return name;
 }
 
 private void setName(String name) {
  this.name = name;
 }
}
在TestData中增加了私有的setName方法,下面是测试代码:
public class TestReflection extends TestCase { 
 public void testInvokePrivateMethod() throws Exception {
  TestData data = new TestData();
  System.out.println(data.getName());
  Assert.assertEquals("1", data.getName());
  Method m = data.getClass().getDeclaredMethod("setName", String.class);
  m.setAccessible(true);
  m.invoke(data, "3");
  System.out.println(data.getName());
  Assert.assertEquals("3", data.getName());
 }
}
运行结果:
1
3
其中最关键的代码行是:
m.setAccessible(true);
它把对象data的setName方法的访问属性设置为public.

那么这样调用私有方法,访问私有属性有什么用处呢?
在实际项目中,我们会使用很多其它第三方的包,有的时候是通过修改源代码完成你想要的功能,有的时候,是因为第三方包中仅仅因为某几个方法的访问属性被设置为private,或者只要修改private的字段值即可.这个时候,用这种反射的方法就可以很容易实现了.

另外一个场景就是从系统架构层来考虑数据封装.例如系统有一些元数据类,99%的情况下,我们只是提供get方法供其它应用层获得字段的值,如果把修改的 set方法也提供出去,那么可能会影响到系统的可维护性.而在系统运行期间,又很难避免的要修改这些元数据的值.这种情况下,也可以通过这种反射的方式来 实现.