[英]Getting value of private not primitive field using reflection
假设我们有两个不同的程序包...一个程序包无法访问,但是我们想知道一个称为b的复杂字段的值。
public class A {
private String whatever;
private B b;
private static class B {
final ArrayList<Z> c = new ArrayList<Z>();
private void addItem(Z z) {
this.c.add(z);
}
private Z getItem(int nr) {
return this.c.get(nr);
}
}
}
public class Reflect extends A {
public static void main(String[] args) throws NoSuchFieldException, SecurityException {
Reflect ref = new Reflect();
Class getA = ref.getClass().getSuperclass();
Field getB = getDeclaredField("b");
getB.setAccessible(true);
Class bInst = getB.getClass();
Method bMeth = bInst.getMethod("getItem", Integer.TYPE);
Object zInst = bMeth.invoke(new Integer(123));
}
}
如果没有从包中获取复杂的B型,该如何获取值? 即使我将字段gstB设置为可访问的,仍然会收到java.lang.NoSuchMethodException:stackOver.A.getItem(int)。
如果没有从包中获取复杂的B型,该如何获取值?
您可以将其作为Object
获取,然后使用反射进一步发现它公开的方法。
Object bInst = ... // Get b through reflection
Class bClass = bInst.getClass();
Method[] bMeth = bClass.getMethod("getItem", Integer.TYPE);
Object zInst = bMeth.invoke(new Integer(123));
您唯一缺少的是getField仅给您提供公共可访问字段。
Field getB = getA.getDeclaredField("b");
将为您提供该班级的任何领域。
一个更长的例子
class Main {
public static class A {
private String whatever;
private B b = new B();
private static class B {
final ArrayList<String> c = new ArrayList<String>();
private void addItem(String z) {
this.c.add(z);
}
private String getItem(int nr) {
return this.c.get(nr);
}
}
}
public static class Reflect extends A {
public static void main(String... ignored) throws Exception {
Reflect ref = new Reflect();
Class getA = ref.getClass().getSuperclass();
Field getB = getA.getDeclaredField("b");
getB.setAccessible(true);
Object b = getB.get(ref);
Method addItem = b.getClass().getDeclaredMethod("addItem", String.class);
addItem.setAccessible(true);
addItem.invoke(b, "Hello");
Method getItem = b.getClass().getDeclaredMethod("getItem", int.class);
getItem.setAccessible(true);
String hi = (String) getItem.invoke(b, 0);
System.out.println(hi);
}
}
}
版画
Hello
使用commons beanutils库并使用以下方法,这比自己做的要干净得多
PropertyUtils.getNestedProperty(ref, "b.propertyOfClassB");
用实际的属性名称替换propertyOfClassB。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.