Найкращий спосіб викликати геттера шляхом відображення


127

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

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

Мені цікаво, чи є кращий спосіб викликати цей геттер (багато фреймворків використовують getters / setters для доступу до атрибутів, так що, можливо, вони роблять іншим способом).

Дякую

Відповіді:


240

Я думаю, це повинно вказувати вам на правильний напрямок:

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 поставляється з іншими зручними речами. тобто перетворення значень на ходу (об'єкт у рядок, рядок в об’єкт) для спрощення налаштування властивостей із введення користувача.


Велике спасибі! Це позбавило мене від жорстких маніпуляцій тощо!
guerda

1
Хороший дзвінок у BeanUtils Apache. Полегшує отримання / налаштування властивостей та обробляє перетворення типів.
Пітер Ценг

Чи є спосіб викликати методи в тому порядку, в якому поля перераховані у файлі Java?
LifeAndHope

Подивіться мою відповідь нижче @Anand
Anand

Дуже сподобалося ! Дивовижно.
smilyface

20

Ви можете використовувати для цього рамку Reflections

import static org.reflections.ReflectionUtils.*;
Set<Method> getters = ReflectionUtils.getAllMethods(someClass,
      withModifier(Modifier.PUBLIC), withPrefix("get"), withAnnotation(annotation));

Будьте уважні, що Роздуми все ще не сумісні з Java 9 . Є посилання на краще поведінку ClassIndex (час компіляції) та ClassGraph (час виконання) альтернативи від threre.
Вадим

Це рішення також не враховуватиметься * getters на відміну від bean Introspector у прийнятій відповіді.
Вадим


3

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

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
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.