Доступ к частным унаследованным полям через отражение в Java - PullRequest
104 голосов
/ 25 августа 2010

Я нашел способ получить унаследованные члены через class.getDeclaredFields(); и доступ к закрытым членам через class.getFields() Но я ищу частные унаследованные поля.Как мне этого добиться?

Ответы [ 7 ]

123 голосов
/ 25 августа 2010

Это должно продемонстрировать, как ее решить:

import java.lang.reflect.Field;

class Super {
    private int i = 5;
}

public class B extends Super {
    public static void main(String[] args) throws Exception {
        B b = new B();
        Field[] fs = b.getClass().getSuperclass().getDeclaredFields();
        fs[0].setAccessible(true);
        System.out.println(fs[0].get(b));
    }
}

Выход:

5
43 голосов
/ 25 августа 2010

Лучшим подходом здесь является использование Шаблон посетителя , который действительно находит все поля в классе и всех суперклассах и выполняет для них действие обратного вызова.


Реализация

Spring имеет хороший класс Utility ReflectionUtils, который делает именно это: он определяет метод для циклического перебора всех полей всех суперклассов с обратным вызовом: ReflectionUtils.doWithFields()

Документация:

Вызвать данный обратный вызов для всех полей в целевом классе, поднимаясь по иерархии классов, чтобы получить все объявленные поля.

Параметры:
- clazz - целевой класс для анализа
- fc - обратный вызов для вызова для каждого поля
- ff - фильтр, определяющий поля для применения обратного вызова к

Пример кода:

ReflectionUtils.doWithFields(RoleUnresolvedList.class,
    new FieldCallback(){

        @Override
        public void doWith(final Field field) throws IllegalArgumentException,
            IllegalAccessException{

            System.out.println("Found field " + field + " in type "
                + field.getDeclaringClass());

        }
    },
    new FieldFilter(){

        @Override
        public boolean matches(final Field field){
            final int modifiers = field.getModifiers();
            // no static fields please
            return !Modifier.isStatic(modifiers);
        }
    });

Выходные данные:

Найдено поле private переходный логический javax.management.relation.RoleUnresolvedList.typeSafe в классе типов javax.management.relation.RoleUnresolvedList
Найдено полечастный временный логический javax.management.relation.RoleUnresolvedList.tainted в классе типов javax.management.relation.RoleUnresolvedList
Найдено поле частного временного java.lang.Object [] java.util.ArrayList.elementData в классе типов java.utilArrayList
Найдено поле private int java.util.ArrayList.size в классе типов java.util.ArrayList
Найдено поле защищенного временного объекта int java.util.AbstractList.modCount в классе типов java.util.AbstractList

33 голосов
/ 25 августа 2010

Это будет сделано:

private List<Field> getInheritedPrivateFields(Class<?> type) {
    List<Field> result = new ArrayList<Field>();

    Class<?> i = type;
    while (i != null && i != Object.class) {
        Collections.addAll(result, i.getDeclaredFields());
        i = i.getSuperclass();
    }

    return result;
}

Если вы используете инструмент покрытия кода, такой как EclEmma , вы должны остерегаться: они добавляют скрытое поле для каждого из ваших классов,В случае EclEmma эти поля помечены синтетические , и вы можете отфильтровать их следующим образом:

private List<Field> getInheritedPrivateFields(Class<?> type) {
    List<Field> result = new ArrayList<Field>();

    Class<?> i = type;
    while (i != null && i != Object.class) {
        for (Field field : i.getDeclaredFields()) {
            if (!field.isSynthetic()) {
                result.add(field);
            }
        }
        i = i.getSuperclass();
    }

    return result;
}
18 голосов
/ 18 апреля 2013
public static Field getField(Class<?> clazz, String fieldName) {
    Class<?> tmpClass = clazz;
    do {
        try {
            Field f = tmpClass.getDeclaredField(fieldName);
            return f;
        } catch (NoSuchFieldException e) {
            tmpClass = tmpClass.getSuperclass();
        }
    } while (tmpClass != null);

    throw new RuntimeException("Field '" + fieldName
            + "' not found on class " + clazz);
}

(на основании этого ответа)

15 голосов
/ 25 августа 2010

На самом деле я использую иерархию сложного типа, поэтому ваше решение не является полным.Мне нужно сделать рекурсивный вызов, чтобы получить все частные унаследованные поля.Вот мое решение

 /**
 * Return the set of fields declared at all level of class hierachy
 */
public Vector<Field> getAllFields(Class clazz) {
    return getAllFieldsRec(clazz, new Vector<Field>());
}

private Vector<Field> getAllFieldsRec(Class clazz, Vector<Field> vector) {
    Class superClazz = clazz.getSuperclass();
    if(superClazz != null){
        getAllFieldsRec(superClazz, vector);
    }
    vector.addAll(toVector(clazz.getDeclaredFields()));
    return vector;
}
8 голосов
/ 09 октября 2013

Мне нужно было добавить поддержку унаследованных полей для чертежей в Модель Citizen . Я получил этот метод, который является более кратким для получения полей класса + унаследованных полей.

private List<Field> getAllFields(Class clazz) {
    List<Field> fields = new ArrayList<Field>();

    fields.addAll(Arrays.asList(clazz.getDeclaredFields()));

    Class superClazz = clazz.getSuperclass();
    if(superClazz != null){
        fields.addAll(getAllFields(superClazz));
    }

    return fields;
}
7 голосов
/ 27 августа 2010
private static Field getField(Class<?> clazz, String fieldName) {
    Class<?> tmpClass = clazz;
    do {
        for ( Field field : tmpClass.getDeclaredFields() ) {
            String candidateName = field.getName();
            if ( ! candidateName.equals(fieldName) ) {
                continue;
            }
            field.setAccessible(true);
            return field;
        }
        tmpClass = tmpClass.getSuperclass();
    } while ( clazz != null );
    throw new RuntimeException("Field '" + fieldName +
        "' not found on class " + clazz);
}
...