परावर्तन द्वारा प्राप्त करने वाले का सर्वोत्तम तरीका


127

मुझे किसी विशिष्ट एनोटेशन के साथ फ़ील्ड का मान प्राप्त करने की आवश्यकता है, इसलिए प्रतिबिंब के साथ मैं इस फ़ील्ड ऑब्जेक्ट को प्राप्त करने में सक्षम हूं। समस्या यह है कि यह क्षेत्र हमेशा निजी होगा, हालांकि मुझे पहले से पता है कि इसमें हमेशा एक विधि होगी। मुझे पता है कि मैं setAccesible (सत्य) का उपयोग कर सकता हूं और इसका मूल्य प्राप्त कर सकता हूं (जब कोई PermissionManager नहीं है), हालांकि मैं इसके गेटर विधि को लागू करना पसंद करता हूं।

मुझे पता है कि मैं "get + fieldName" की तलाश करके विधि की तलाश कर सकता हूं (हालांकि मुझे बूलियन क्षेत्रों के लिए उदाहरण के लिए पता है कि कभी-कभी "+ फ़ील्डनाम" नाम दिया जाता है)।

मुझे आश्चर्य है कि अगर इस गेट्टर को लागू करने का एक बेहतर तरीका है (कई चौखटे विशेषताओं का उपयोग करने के लिए गेटर्स / सेटर का उपयोग करते हैं तो शायद वे दूसरे तरीके से करते हैं)।

धन्यवाद

जवाबों:


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));
}

ध्यान दें कि आप Introspector का उपयोग किए बिना अपने आप को BeanInfo या PropertyDescriptor इंस्टेंस बना सकते हैं। हालांकि, इंट्रोस्पेक्टर आंतरिक रूप से कुछ कैशिंग करता है जो सामान्य रूप से एक अच्छा थिंग (tm) है। यदि आप कैश के बिना खुश हैं, तो आप भी जा सकते हैं

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

हालाँकि, बहुत सारे पुस्तकालय हैं जो java.beans API का विस्तार और सरलीकरण करते हैं। कॉमन्स बीन्यूटिल्स एक प्रसिद्ध उदाहरण है। वहाँ, आप बस करेंगे:

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

बीन्यूटिल्स अन्य उपयोगी सामान के साथ आता है। उपयोगकर्ता इनपुट से गुणों को सरल बनाने के लिए यानी ऑन-द-फ्लाई मान रूपांतरण (ऑब्जेक्ट टू स्ट्रिंग, स्ट्रिंग टू ऑब्जेक्ट)।


आपका बहुत बहुत धन्यवाद! यह मुझे स्ट्रिंग जोड़तोड़ आदि से बख्शा!
गुर्जरा

1
Apache के BeanUtils पर अच्छा कॉल। संपत्तियों की प्राप्ति / स्थापना को आसान बनाता है, और प्रकार रूपांतरण को संभालता है।
पीटर Tseng

क्या जावा फ़ाइल में फ़ील्ड को सूचीबद्ध करने के क्रम में विधियों को लागू करने का एक तरीका है?
LifeAndHope

मेरे जवाब को नीचे देखें @
आनंद

पसन्द आया ! बहुत बढ़िया।
स्माइलफेस

20

आप इसके लिए रिफ्लेक्शंस फ्रेमवर्क का उपयोग कर सकते हैं

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

सावधान रहें कि प्रतिबिंब अभी भी जावा 9 के साथ संगत नहीं हैं । बेहतर व्यवहार कक्षा के लिए लिंक हैं (संकलित समय) और सिंहासन से क्लासग्राफ (रन समय) विकल्प।
वडज़िम

यह समाधान भी ध्यान में नहीं है * स्वीकृत उत्तर में बीन इंट्रोस्पेक्टर के विपरीत * गेटर्स है।
वडज़िम


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 Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.