用反射设置私有字段值


76

我有2个班级:FatherChild

public class Father implements Serializable, JSONInterface {

    private String a_field;

    //setter and getter here

}

public class Child extends Father {
    //empty class
}

通过反思,我想a_fieldChild课堂上设置:

Class<?> clazz = Class.forName("Child");
Object cc = clazz.newInstance();

Field f1 = cc.getClass().getField("a_field");
f1.set(cc, "reflecting on life");
String str1 = (String) f1.get(cc.getClass());
System.out.println("field: " + str1);

但我有一个例外:

线程“主”中的异常java.lang.NoSuchFieldException:a_field

但是,如果我尝试:

Child child = new Child();
child.setA_field("123");

有用。

使用setter方法,我有同样的问题:

method = cc.getClass().getMethod("setA_field");
method.invoke(cc, new Object[] { "aaaaaaaaaaaaaa" });

2
您需要先获得Class的超类,然后才能访问该字段。另一个SO问题
SomeJavaGuy 2015年

1
该字段不是Child类的一部分,因为它是私有的,所以在Child类中甚至都不可见。
f1sh

与其检索私有字段,
不如

是的,这是一个很好的解决方案。现在我可以尝试
user1066183

Answers:


141

要访问私有字段,您需要设置Field::setAccessible为true。您可以离开超类领域。此代码有效:

Class<?> clazz = Child.class;
Object cc = clazz.newInstance();

Field f1 = cc.getClass().getSuperclass().getDeclaredField("a_field");
f1.setAccessible(true);
f1.set(cc, "reflecting on life");
String str1 = (String) f1.get(cc);
System.out.println("field: " + str1);

75

使用FieldUtils来自Apache的百科全书郎3

FieldUtils.writeField(childInstance, "a_field", "Hello", true);

true即使字段是private,也要设置的强制力。


谢天谢地,这是必须由Spring Bean依赖注入设置的那些愚蠢字段的转义。
Sridhar Sarnobat,

4

这个人也可以访问私有字段,而无需执行任何操作

import org.apache.commons.lang3.reflect.FieldUtils;
Object value = FieldUtils.readField(entity, fieldName, true);

3

Kotlin版本

使用以下扩展功能获取私有变量

fun <T : Any> T.getPrivateProperty(variableName: String): Any? {
    return javaClass.getDeclaredField(variableName).let { field ->
        field.isAccessible = true
        return@let field.get(this)
    }
}

设置私有变量值获取变量

fun <T : Any> T.setAndReturnPrivateProperty(variableName: String, data: Any): Any? {
    return javaClass.getDeclaredField(variableName).let { field ->
        field.isAccessible = true
        field.set(this, data)
        return@let field.get(this)
    }
}

获取变量用途:

val bool = <your_class_object>.getPrivateProperty("your_variable") as String

设置并获取变量用途:

val bool = <your_class_object>.setAndReturnPrivateProperty("your_variable", true) as Boolean
val str = <your_class_object>.setAndReturnPrivateProperty("your_variable", "Hello") as String

Java版本

public class RefUtil {

    public static Field setFieldValue(Object object, String fieldName, Object valueTobeSet) throws NoSuchFieldException, IllegalAccessException {
        Field field = getField(object.getClass(), fieldName);
        field.setAccessible(true);
        field.set(object, valueTobeSet);
        return field;
    }

    public static Object getPrivateFieldValue(Object object, String fieldName) throws NoSuchFieldException, IllegalAccessException {
        Field field = getField(object.getClass(), fieldName);
        field.setAccessible(true);
        return field.get(object);
    }

    private static Field getField(Class mClass, String fieldName) throws NoSuchFieldException {
        try {
            return mClass.getDeclaredField(fieldName);
        } catch (NoSuchFieldException e) {
            Class superClass = mClass.getSuperclass();
            if (superClass == null) {
                throw e;
            } else {
                return getField(superClass, fieldName);
            }
        }
    }
}

设定私人价值用途

RefUtil.setFieldValue(<your_class_object>, "your_variableName", newValue);

获得私人价值使用

Object value = RefUtil.getPrivateFieldValue(<your_class_object>, "your_variableName");

1

您可以使用Manifold的 @Jailbreak进行直接的类型安全的Java反射:

@Jailbreak Child child = new Child();
child.a_field = "123;

@Jailbreak 在编译器中解锁子局部变量,以直接访问其中的所有成员 Child层次结构中的。

同样,您可以将jailbreak()扩展方法一次性使用:

child.jailbreak().a_field = "123";

通过该jailbreak()方法,您可以访问中的任何成员Child层次结构中的。

在这两种情况下,编译器都会像公共字段一样安全地为您解析字段访问,而Manifold会在后台为您生成有效的反射代码。

发现有关歧管的更多信息。


-5

根据Class.getField(强调我的)Javadoc :

返回一个Field对象,该对象反映此Class对象表示的类或接口的指定公共成员字段

此方法仅返回公共字段。以来a_field是私人的,因此找不到。

这是一个工作代码:

public class Main {

    public static void main(String[] args) throws Exception {
        Class<?> clazz = Class.forName("Child");
        Object cc = clazz.newInstance();

        Field f1 = cc.getClass().getField("a_field");
        f1.set(cc, "reflecting on life");
        String str1 = (String) f1.get(cc);
        System.out.println("field: " + str1);
    }

}

class Father implements Serializable {
    public String a_field;
}

class Child extends Father {
//empty class
}

请注意,我还将行更改String str1 = (String) f1.get(cc.getClass());为,String str1 = (String) f1.get(cc);因为您需要提供字段的对象,而不是类。


如果要保留字段的私有性,则需要检索getter / setter方法并调用它们。您提供的代码不起作用,因为要获取方法,还需要指定其参数,因此

cc.getClass().getMethod("setA_field");

一定是

cc.getClass().getMethod("setA_field", String.class);

这是一个工作代码:

public class Main {

    public static void main(String[] args) throws Exception {
        Class<?> clazz = Class.forName("Child");
        Object cc = clazz.newInstance();
        cc.getClass().getMethod("setA_field", String.class).invoke(cc, "aaaaaaaaaaaaaa");
        String str1 = (String) cc.getClass().getMethod("getA_field").invoke(cc);
        System.out.println("field: " + str1);
    }

}

class Father implements Serializable {

    private String a_field;

    public String getA_field() {
        return a_field;
    }

    public void setA_field(String a_field) {
        this.a_field = a_field;
    }

}

class Child extends Father {
    //empty class
}

谢谢..但是a_field必须是私有的。或者,我可以调用setter方法吗?
user1066183
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.