Лучший способ вызвать геттера отражением - PullRequest
119 голосов
/ 14 апреля 2010

Мне нужно получить значение поля с конкретной аннотацией, поэтому с помощью отражения я могу получить этот объект поля. Проблема в том, что это поле всегда будет закрытым, хотя я заранее знаю, что у него всегда будет метод получения. Я знаю, что могу использовать setAccesible (true) и получить его значение (когда PermissionManager отсутствует), хотя я предпочитаю вызывать его метод получения.

Я знаю, что мог бы искать метод, ища "get + fieldName" (хотя я знаю, например, что для логических полей иногда называют "is + fieldName").

Интересно, есть ли лучший способ вызвать этот метод получения (многие фреймворки используют методы получения / установки для доступа к атрибутам, так что, возможно, они делают это по-другому).

Спасибо

Ответы [ 4 ]

223 голосов
/ 14 апреля 2010

Я думаю, что это должно указать вам правильное направление:

import java.beans.*

for (PropertyDescriptor pd : Introspector.getBeanInfo(Foo.class).getPropertyDescriptors()) {
  if (pd.getReadMethod() != null && !"class".equals(pd.getName()))
    System.out.println(pd.getReadMethod().invoke(foo));
}

Обратите внимание, что вы можете создавать экземпляры BeanInfo или PropertyDescriptor самостоятельно, т.е. без использования Introspector.Тем не менее, Introspector выполняет некоторое внутреннее кэширование, что обычно является хорошей вещью (tm).Если вы счастливы без кеша, вы можете даже пойти на

// TODO check for non-existing readMethod
Object value = new PropertyDescriptor("name", Person.class).getReadMethod().invoke(person);

Однако существует множество библиотек, которые расширяют и упрощают API java.beans.Commons BeanUtils является хорошо известным примером.Там вы просто сделаете:

Object value = PropertyUtils.getProperty(person, "name");

BeanUtils поставляется с другими полезными вещами.т.е. преобразование значений «на лету» (объект в строку, строка в объект) для упрощения установки свойств из пользовательского ввода.

20 голосов
/ 02 декабря 2013

Вы можете использовать Reflections framework для этого

import static org.reflections.ReflectionUtils.*;
Set<Method> getters = ReflectionUtils.getAllMethods(someClass,
      withModifier(Modifier.PUBLIC), withPrefix("get"), withAnnotation(annotation));
4 голосов
/ 14 апреля 2010

Соглашение об именах является частью устоявшейся спецификации JavaBeans и поддерживается классами в java.beans пакете.

2 голосов
/ 30 июля 2015

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

public class Student {

    private String grade;

    private String name;

    private String id;

    private String gender;

    private Method[] methods;

    @Retention(RetentionPolicy.RUNTIME)
    public @interface Order {
        int value();
    }

    /**
     * Sort methods as per Order Annotations
     * 
     * @return
     */
    private void sortMethods() {

        methods = Student.class.getMethods();

        Arrays.sort(methods, new Comparator<Method>() {
            public int compare(Method o1, Method o2) {
                Order or1 = o1.getAnnotation(Order.class);
                Order or2 = o2.getAnnotation(Order.class);
                if (or1 != null && or2 != null) {
                    return or1.value() - or2.value();
                }
                else if (or1 != null && or2 == null) {
                    return -1;
                }
                else if (or1 == null && or2 != null) {
                    return 1;
                }
                return o1.getName().compareTo(o2.getName());
            }
        });
    }

    /**
     * Read Elements
     * 
     * @return
     */
    public void readElements() {
        int pos = 0;
        /**
         * Sort Methods
         */
        if (methods == null) {
            sortMethods();
        }
        for (Method method : methods) {
            String name = method.getName();
            if (name.startsWith("get") && !name.equalsIgnoreCase("getClass")) {
                pos++;
                String value = "";
                try {
                    value = (String) method.invoke(this);
                }
                catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
                    e.printStackTrace();
                }
                System.out.println(name + " Pos: " + pos + " Value: " + value);
            }
        }
    }

    // /////////////////////// Getter and Setter Methods

    /**
     * @param grade
     * @param name
     * @param id
     * @param gender
     */
    public Student(String grade, String name, String id, String gender) {
        super();
        this.grade = grade;
        this.name = name;
        this.id = id;
        this.gender = gender;
    }

    /**
     * @return the grade
     */
    @Order(value = 4)
    public String getGrade() {
        return grade;
    }

    /**
     * @param grade the grade to set
     */
    public void setGrade(String grade) {
        this.grade = grade;
    }

    /**
     * @return the name
     */
    @Order(value = 2)
    public String getName() {
        return name;
    }

    /**
     * @param name the name to set
     */
    public void setName(String name) {
        this.name = name;
    }

    /**
     * @return the id
     */
    @Order(value = 1)
    public String getId() {
        return id;
    }

    /**
     * @param id the id to set
     */
    public void setId(String id) {
        this.id = id;
    }

    /**
     * @return the gender
     */
    @Order(value = 3)
    public String getGender() {
        return gender;
    }

    /**
     * @param gender the gender to set
     */
    public void setGender(String gender) {
        this.gender = gender;
    }

    /**
     * Main
     * 
     * @param args
     * @throws IOException
     * @throws SQLException
     * @throws InvocationTargetException
     * @throws IllegalArgumentException
     * @throws IllegalAccessException
     */
    public static void main(String args[]) throws IOException, SQLException, IllegalAccessException,
            IllegalArgumentException, InvocationTargetException {
        Student student = new Student("A", "Anand", "001", "Male");
        student.readElements();
    }
  }

Вывод при сортировке

getId Pos: 1 Value: 001
getName Pos: 2 Value: Anand
getGender Pos: 3 Value: Male
getGrade Pos: 4 Value: A
...