使用getDeclaredField()和字段时存在NoSuchFieldException

时间:2015-02-14 16:04:48

标签: java reflection

  

java.lang.NoSuchFieldException:c
  at java.lang.Class.getDeclaredField(Unknown Source)
  在ru.onlymc.OnlyMZ.CustomEntityType.getPrivateStatic(CustomEntityType.java:177)

方式:

private static Object getPrivateStatic(Class clazz, String f) throws Exception {
    Field field = clazz.getDeclaredField(f);
    field.setAccessible(true);
    return field.get(null);
}

调用

private static void a(Class paramClass, String paramString, int paramInt) {
    try {
        ((Map) getPrivateStatic(sg.class, "c")).put(paramString, paramClass);
        //...
    } catch (Exception exc) {
        exc.printStackTrace();
    }
}

sg.class (从反编译到确定必填字段确实存在)

private static Map c = new HashMap();
private static Map d = new HashMap();
private static Map e = new HashMap();
private static Map f = new HashMap();
private static Map g = new HashMap();
public static HashMap a = new LinkedHashMap();

2 个答案:

答案 0 :(得分:0)

抱歉,我无法复制此内容。

这是我运行的完整源代码:

import java.lang.reflect.*;
import java.util.*;

@SuppressWarnings("unchecked")
public class ReflectionTest {

    private static Object getPrivateStatic(Class clazz, String f) throws Exception {
        Field field = clazz.getDeclaredField(f);
        field.setAccessible(true);
        return field.get(null);
    }

    private static void a(Class paramClass, String paramString, int paramInt) {
        try {
            ((Map) getPrivateStatic(sg.class, "c")).put(paramString, paramClass);
            //...
        } catch (Exception exc) {
            exc.printStackTrace();
        }
    }

    public static void main(String[] args) throws Exception {
        a(String.class, "test", 0);
        sg.printC();
    }
}

class sg {
    private static Map c = new HashMap();
    private static Map d = new HashMap();
    private static Map e = new HashMap();
    private static Map f = new HashMap();
    private static Map g = new HashMap();
    public static HashMap a = new LinkedHashMap();

    public static void printC() {
        System.out.println(c);
    }
}

这是我运行时得到的输出:

{test=class java.lang.String}

鉴于您尚未指定sg类的完整反编译源,我只能猜测可能发生的一些事情:

  • 有多个名为sg的类,您的代码正在使用其中一个,但您的反编译输出来自另一个。
  • 这些字段存在于sg内的内部类中。

编辑:您链接到下面的班级sg似乎包含静态字段c,并且不包含内部类,所以我希望能够使用反射访问此字段。我不能将sg类与我上面写的ReflectionTest类一起使用,因为它依赖于许多其他具有模糊名称的类,例如xk

我只能得出结论,您对于您尝试从哪个类中访问字段c感到困惑。我建议将getPrivateStatic方法更改为以下内容,这可能会提供更有用的错误消息,包括类的名称及其中的所有字段:

    private static Object getPrivateStatic(Class clazz, String f) throws Exception {
        try {
            Field field = clazz.getDeclaredField(f);
            field.setAccessible(true);
            return field.get(null);
        }
        catch (NoSuchFieldException e) {
            // Throw a more helpful exception.
            throw new NoSuchFieldException(
                "Could not find field named '" + f + "' in class '" + clazz +
                "'.  All fields: " + Arrays.asList(clazz.getDeclaredFields()));
        }
    }

答案 1 :(得分:-1)

您正试图将对象的字段设为null。请参阅Field.get(Object)

上的文档

您需要提供一个对象来获取其字段内容。或者您需要提供一个类来获取静态字段内容。

所以你应该写下面的内容:

private static Object getPrivateStatic(Class clazz, String f) throws Exception {
  Field field = clazz.getDeclaredField(f);
  field.setAccessible(true);
  return field.get(clazz);
}

顺便说一句。在生产代码中使用反射不被认为是好的编程风格,因为它例如是使重构成问题。

如果您仍然需要使用反射(或者不使用生产代码),请使用这种东西的框架(访问私有静态字段)。用例如PrivilegedAccessor这种方法将是一个单行程序:

PA.getValue(clazz, f);