जावा प्रतिबिंब का उपयोग कर विरासत में मिली विशेषता नामों / मूल्यों को पुनः प्राप्त करना


128

मेरे पास जावा ऑब्जेक्ट 'चाइल्डऑब्ज' है जो 'पैरेंटबोज' से बढ़ा है। अब, यदि चाइल्डऑनज के सभी विशेषता नाम और मूल्यों को पुनः प्राप्त करना संभव है, तो विरासत में मिली विशेषताओं सहित, जावा प्रतिबिंब तंत्र का उपयोग करना भी शामिल है?

Class.getFields मुझे सार्वजनिक विशेषताओं की श्रेणी देता है , और Class.getDeclaredFields मुझे सभी फ़ील्ड्स की सरणी देता है, लेकिन उनमें से कोई भी विरासत में मिली फ़ील्ड सूची शामिल नहीं करता है।

क्या विरासत में मिली विशेषताओं को पुनः प्राप्त करने का कोई तरीका है?

जवाबों:


173

नहीं, आपको इसे स्वयं लिखने की आवश्यकता है। यह एक सरल पुनरावर्ती विधि है जिसे Class.getSuperClass () पर कहा जाता है :

public static List<Field> getAllFields(List<Field> fields, Class<?> type) {
    fields.addAll(Arrays.asList(type.getDeclaredFields()));

    if (type.getSuperclass() != null) {
        getAllFields(fields, type.getSuperclass());
    }

    return fields;
}

@Test
public void getLinkedListFields() {
    System.out.println(getAllFields(new LinkedList<Field>(), LinkedList.class));
}

2
हाँ। उसके बारे में सोचा। लेकिन अगर कोई अन्य तरीका है कि जाँच करना चाहता था। धन्यवाद। :)
वीरा

7
में एक परस्पर तर्क पारित करना और इसे वापस करना शायद एक महान डिजाइन नहीं है। fields.addAll (type.getDeclaredFields ()); ऐड के साथ लूप के लिए एन्हांस की तुलना में अधिक पारंपरिक होगा।
टॉम हॉकिन -

मुझे कम से कम इसे संकलित करने की आवश्यकता महसूस होगी (स्टैकओवरफ़्लो पर!), और शायद थोड़े Arrays.asList में जोड़ें।
टॉम हॉन्टिन -

ऐसा लगता है कि आपका कोड सभी क्षेत्रों को इकट्ठा करता है, साथ ही निजी और स्थिर क्षेत्र जो विरासत में नहीं मिले हैं।
पीटर वेरहास

90
    public static List<Field> getAllFields(Class<?> type) {
        List<Field> fields = new ArrayList<Field>();
        for (Class<?> c = type; c != null; c = c.getSuperclass()) {
            fields.addAll(Arrays.asList(c.getDeclaredFields()));
        }
        return fields;
    }

9
यह मेरा पसंदीदा समाधान है, हालांकि मैं इसे "getAllFields" कहूंगा क्योंकि यह दिए गए वर्ग के क्षेत्रों को भी लौटाता है।
पीनो

3
हालांकि मुझे बहुत अधिक पुनरावृत्ति पसंद है (यह मजेदार है!), मैं इस विधि की पठनीयता और अधिक सहज ज्ञान युक्त मापदंडों (पास होने के लिए एक नया संग्रह की आवश्यकता नहीं) को पसंद करता हूं, और नहीं तो (खंड में निहित) और खेतों पर कोई पुनरावृत्ति नहीं खुद को।
रेमी मोरिन

यह दिखाता है कि पुनरावर्ती अनावश्यक है और .. मुझे छोटे कोड पसंद हैं! धन्यवाद! :)
कुंभ राशि पावर

कई वर्षों में मुझे हमेशा लगता है कि इसके लिए प्रारंभिक मूल्य सिर्फ एक पूर्णांक है, @ वीरा के सवाल से मुझे लगता है कि केवल पुनरावर्ती इसे हल कर सकते हैं, @ एस्को लुओंटोला आपकी आज्ञा भयानक है।
तोया अकीरा

@ ईस्को: बहुत बहुत धन्यवाद। दिन बचा लिया! यह संक्षिप्त है और निर्दोष रूप से काम करता है!
गौरव

37

यदि इसके बजाय आप इसे पूरा करने के लिए किसी लाइब्रेरी पर निर्भर रहना चाहते हैं, तो Apache Commons Lang संस्करण 3.2+ प्रदान करता है FieldUtils.getAllFieldsList:

import java.lang.reflect.Field;
import java.util.AbstractCollection;
import java.util.AbstractList;
import java.util.AbstractSequentialList;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;

import org.apache.commons.lang3.reflect.FieldUtils;
import org.junit.Assert;
import org.junit.Test;

public class FieldUtilsTest {

    @Test
    public void testGetAllFieldsList() {

        // Get all fields in this class and all of its parents
        final List<Field> allFields = FieldUtils.getAllFieldsList(LinkedList.class);

        // Get the fields form each individual class in the type's hierarchy
        final List<Field> allFieldsClass = Arrays.asList(LinkedList.class.getFields());
        final List<Field> allFieldsParent = Arrays.asList(AbstractSequentialList.class.getFields());
        final List<Field> allFieldsParentsParent = Arrays.asList(AbstractList.class.getFields());
        final List<Field> allFieldsParentsParentsParent = Arrays.asList(AbstractCollection.class.getFields());

        // Test that `getAllFieldsList` did truly get all of the fields of the the class and all its parents 
        Assert.assertTrue(allFields.containsAll(allFieldsClass));
        Assert.assertTrue(allFields.containsAll(allFieldsParent));
        Assert.assertTrue(allFields.containsAll(allFieldsParentsParent));
        Assert.assertTrue(allFields.containsAll(allFieldsParentsParentsParent));
    }
}

6
बूम! मुझे पहिए को फिर से लगाना पसंद नहीं है। इसके लिए चीयर्स करें।
जोशुआ पिंटर

6

आपको कॉल करने की आवश्यकता है:

Class.getSuperclass().getDeclaredFields()

आवश्यक के रूप में विरासत पदानुक्रम को दोहराते हुए।



4

पुनरावर्ती समाधान ठीक हैं, एकमात्र छोटा मुद्दा यह है कि वे घोषित और विरासत वाले सदस्यों का एक सुपरसेट लौटाते हैं। ध्यान दें कि getDeclaredFields () विधि रिटर्न भी निजी तरीके हैं। तो यह देखते हुए कि आप पूरे सुपरक्लास पदानुक्रम को नेविगेट करते हैं, आप सुपरक्लास में घोषित सभी निजी क्षेत्रों को शामिल करेंगे, और जो विरासत में नहीं मिलते हैं।

एक Modifier.isPublic के साथ एक साधारण फिल्टर || Modifier.isProtected विधेय करेगा:

import static java.lang.reflect.Modifier.isPublic;
import static java.lang.reflect.Modifier.isProtected;

(...)

List<Field> inheritableFields = new ArrayList<Field>();
for (Field field : type.getDeclaredFields()) {
    if (isProtected(field.getModifiers()) || isPublic(field.getModifiers())) {
       inheritableFields.add(field);
    }
}

2
private static void addDeclaredAndInheritedFields(Class<?> c, Collection<Field> fields) {
    fields.addAll(Arrays.asList(c.getDeclaredFields())); 
    Class<?> superClass = c.getSuperclass(); 
    if (superClass != null) { 
        addDeclaredAndInheritedFields(superClass, fields); 
    }       
}

ऊपर दिए गए "DidYouMeanThatTomHa ..." समाधान का कार्य संस्करण


2

वसंत उपयोग पुस्तकालय के साथ, आप यह जांचने के लिए उपयोग कर सकते हैं कि क्या कोई विशिष्ट विशेषता कक्षा में मौजूद है:

Field field = ReflectionUtils.findRequiredField(YOUR_CLASS.class, "ATTRIBUTE_NAME");

log.info(field2.getName());

Api doc:
https://docs.spring.io/spring-data/commons/docs/current/api/org/springframework/data/util/ReflectionUtils.html

या

 Field field2 = ReflectionUtils.findField(YOUR_CLASS.class, "ATTRIBUTE_NAME");

 log.info(field2.getName());

Api doc:
https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/util/ReflectionUtils.html

@cheers



1

छोटा और कम वस्तु के साथ त्वरित? ^^

private static Field[] getAllFields(Class<?> type) {
    if (type.getSuperclass() != null) {
        return (Field[]) ArrayUtils.addAll(getAllFields(type.getSuperclass()), type.getDeclaredFields());
    }
    return type.getDeclaredFields();
}

HI @ अलेक्सिस लेग्रोस: एरेयूटिल्स को प्रतीक नहीं मिल सकता।
तोया अकीरा

1
यह वर्ग अपाचे कॉमन्स लैंग का है।
एलेक्सिस लेगरोज

अपाचे में पहले से ही इस प्रश्न के अनुरोध को संभालने के लिए एक FieldUtils.getAllFields फ़ंक्शन है।
तोया अकीरा

1

getFields (): सभी सार्वजनिक क्षेत्रों को पूरी श्रेणी पदानुक्रम तक प्राप्त कर
लेता है और getDeclaredFields (): सभी फ़ील्ड हो जाता है, भले ही उनके संशोधक हों, लेकिन केवल वर्तमान वर्ग के लिए। तो, आपको सभी पदानुक्रम में शामिल होना होगा।
मैंने हाल ही में इस कोड को org.apache.commons.lang3.reflect.ieldUtils से देखा

public static List<Field> getAllFieldsList(final Class<?> cls) {
        Validate.isTrue(cls != null, "The class must not be null");
        final List<Field> allFields = new ArrayList<>();
        Class<?> currentClass = cls;
        while (currentClass != null) {
            final Field[] declaredFields = currentClass.getDeclaredFields();
            Collections.addAll(allFields, declaredFields);
            currentClass = currentClass.getSuperclass();
        }
        return allFields;
}

0
private static void addDeclaredAndInheritedFields(Class c, Collection<Field> fields) {
    fields.addAll(Arrays.asList(c.getDeclaredFields()));
    Class superClass = c.getSuperclass();
    if (superClass != null) {
        addDeclaredAndInheritedFields(superClass, fields);
    }
}

0

यह @ user1079877 द्वारा स्वीकार किए गए उत्तर की एक रिकॉर्डिंग है। यह हो सकता है कि एक संस्करण जो फ़ंक्शन के पैरामीटर को संशोधित नहीं करता है और कुछ आधुनिक जावा सुविधाओं का भी उपयोग करता है।

public <T> Field[] getFields(final Class<T> type, final Field... fields) {
    final Field[] items = Stream.of(type.getDeclaredFields(), fields).flatMap(Stream::of).toArray(Field[]::new);
    if (type.getSuperclass() == null) {
        return items;
    } else {
        return getFields(type.getSuperclass(), items);
    }
}

यह कार्यान्वयन भी आह्वान को थोड़ा और संक्षिप्त बनाता है:

var fields = getFields(MyType.class);

0

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

/**
 * Return a list containing all declared fields and all inherited fields for the given input
 * (but avoiding any quirky enum fields and tool injected fields).
 */
public List<Field> getAllFields(Object input) {
    return getFieldsAndInheritedFields(new ArrayList<>(), input.getClass());
}

private List<Field> getFieldsAndInheritedFields(List<Field> fields, Class<?> inputType) {
    fields.addAll(getFilteredDeclaredFields(inputType));
    return inputType.getSuperclass() == null ? fields : getFieldsAndInheritedFields(fields, inputType.getSuperclass());

}

/**
 * Where the input is NOT an {@link Enum} type then get all declared fields except synthetic fields (ie instrumented
 * additional fields). Where the input IS an {@link Enum} type then also skip the fields that are all the
 * {@link Enum} instances as this would lead to an infinite loop if the user of this class is traversing
 * an object graph.
 */
private List<Field> getFilteredDeclaredFields(Class<?> inputType) {
    return Arrays.asList(inputType.getDeclaredFields()).stream()
                 .filter(field -> !isAnEnum(inputType) ||
                         (isAnEnum(inputType) && !isSameType(field, inputType)))
                 .filter(field -> !field.isSynthetic())
                 .collect(Collectors.toList());

}

private boolean isAnEnum(Class<?> type) {
    return Enum.class.isAssignableFrom(type);
}

private boolean isSameType(Field input, Class<?> ownerType) {
    return input.getType().equals(ownerType);
}

Spock में टेस्ट क्लास (और ग्रूवी सिंथेटिक फील्ड्स को जोड़ता है):

class ReflectionUtilsSpec extends Specification {

    def "declared fields only"() {

        given: "an instance of a class that does not inherit any fields"
        def instance = new Superclass()

        when: "all fields are requested"
        def result = new ReflectionUtils().getAllFields(instance)

        then: "the fields declared by that instance's class are returned"
        result.size() == 1
        result.findAll { it.name in ['superThing'] }.size() == 1
    }


    def "inherited fields"() {

        given: "an instance of a class that inherits fields"
        def instance = new Subclass()

        when: "all fields are requested"
        def result = new ReflectionUtils().getAllFields(instance)

        then: "the fields declared by that instance's class and its superclasses are returned"
        result.size() == 2
        result.findAll { it.name in ['subThing', 'superThing'] }.size() == 2

    }

    def "no fields"() {
        given: "an instance of a class with no declared or inherited fields"
        def instance = new SuperDooperclass()

        when: "all fields are requested"
        def result = new ReflectionUtils().getAllFields(instance)

        then: "the fields declared by that instance's class and its superclasses are returned"
        result.size() == 0
    }

    def "enum"() {

        given: "an instance of an enum"
        def instance = Item.BIT

        when: "all fields are requested"
        def result = new ReflectionUtils().getAllFields(instance)

        then: "the fields declared by that instance's class and its superclasses are returned"
        result.size() == 3
        result.findAll { it.name == 'smallerItem' }.size() == 1
    }

    private class SuperDooperclass {
    }

    private class Superclass extends SuperDooperclass {
        private String superThing
    }


    private class Subclass extends Superclass {
        private String subThing
    }

    private enum Item {

        BIT("quark"), BOB("muon")

        Item(String smallerItem) {
            this.smallerItem = smallerItem
        }

        private String smallerItem

    }
}
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.