All Downloads are FREE. Search and download functionalities are using the official Maven repository.

de.team33.patterns.reflect.luna.Fields Maven / Gradle / Ivy

package de.team33.patterns.reflect.luna;

import de.team33.patterns.exceptional.dione.Converter;
import de.team33.patterns.exceptional.dione.XConsumer;
import de.team33.patterns.exceptional.dione.XFunction;

import java.lang.reflect.Field;
import java.lang.reflect.Member;
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import java.util.function.BiConsumer;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/**
 * Represents an aggregate of {@link Field}s of a specific class and allows elementary operations over all these
 * {@link Field}s, with any {@link IllegalAccessException}s that may occur being encapsulated in unchecked exceptions.
 */
public class Fields {

    private static final int SYNTHETIC = 0x00001000;
    private static final int NOT_SIGNIFICANT = Modifier.STATIC | Modifier.TRANSIENT | SYNTHETIC;
    private static final Converter CNV = Converter.using(AccessException::new);

    private final List fields;

    private Fields(final Group group, final Class subjectClass, final Collection ignorable) {
        this.fields = group.streaming.apply(subjectClass)
                                     .filter(Fields::isSignificant)
                                     .filter(field -> !ignorable.contains(field.getName()))
                                     .collect(Collectors.toList());
    }

    private static boolean isSignificant(final Member field) {
        return isSignificant(field.getModifiers());
    }

    private static boolean isSignificant(final int modifiers) {
        return 0 == (modifiers & NOT_SIGNIFICANT);
    }

    /**
     * Returns a new instance that includes all significant* {@link Field}s that exist in the given class
     * and belong to the given {@link Group}.
     * 

* *Significant fields are those fields that are not declared static, transient, * or synthetic**. * These are typically the fields that make up the "value" of a data object and as such should be considered when * implementing {@code equals()}, {@code hashCode()} and {@code toString()}. *

* **A synthetically declared field, for example, appears in a non-static inner class, * namely a reference to the instance of the outer class. * * @see #of(Class, String...) */ public static Fields of(final Group group, final Class subjectClass, final String... ignorable) { return new Fields(group, subjectClass, Arrays.asList(ignorable)); } /** * Returns a new instance that includes all significant* {@link Field}s declared by the given class. *

* *Significant fields are those fields that are not declared static, transient, * or synthetic**. * These are typically the fields that make up the "value" of a data object and as such should be considered when * implementing {@code equals()}, {@code hashCode()} and {@code toString()}. *

* **A synthetically declared field, for example, appears in a non-static inner class, * namely a reference to the instance of the outer class. * * @see #of(Group, Class, String...) */ public static Fields of(final Class subjectClass, final String... ignorable) { return of(Group.DECLARED, subjectClass, ignorable); } private static BiConsumer, ? super Field> put(final Function function) { return (map, field) -> map.put(field.getName(), function.apply(field)); } /** * Performs a given operation on all contained fields. * If the operation causes an {@link IllegalAccessException}, it's caught, wrapped as an unchecked * {@link AccessException}, and rethrown. *

* In particular, this can be used to implement a copy operation or a copy constructor of a data object class, * example: *

     * public class DataObject {
     *
     *     private static final Fields FIELDS = Fields.of(DataObject.class);
     *
     *     // declare some instance fields ...
     *
     *     public DataObject(final DataObject source) {
     *         FIELDS.forEach(field -> field.set(this, field.get(source)));
     *     }
     *
     *     // ...
     * }
     * 
*

* Note: Access to the value of a field using {@link Field#get(Object)} and {@link Field#set(Object, Object)} * is also possible without any problems for non-public fields if the accessing code resides in the context * of the field's declaring class. * * @throws AccessException If an {@link IllegalAccessException} is thrown during the given operation. */ public final void forEach(final XConsumer operation) throws AccessException { fields.forEach(CNV.consumer(operation)); } /** * Returns a {@link Stream} of the contained fields. */ @SuppressWarnings("BoundedWildcard") public final Stream stream() { return fields.stream(); } /** * Returns a {@link Stream} of the results of a given mapping operation performed on each contained field. * If the operation throws an {@link IllegalAccessException}, it's caught, wrapped as an {@link AccessException} * (an unchecked exception), and rethrown. * * @throws AccessException If an {@link IllegalAccessException} is thrown during the operation. */ @SuppressWarnings("BoundedWildcard") public final Stream map(final XFunction operation) throws AccessException { return stream().map(CNV.function(operation)); } /** * Returns a {@link Map} composed of the names of the fields involved as keys and the results of the given * mapping operation performed on each of the fields as values. * If the operation throws an {@link IllegalAccessException}, it's caught, wrapped as an {@link AccessException} * (an unchecked exception), and rethrown. * * @throws AccessException If an {@link IllegalAccessException} is thrown during the operation. */ public final Map toMap(final XFunction operation) throws AccessException { return stream().collect(TreeMap::new, put(CNV.function(operation)), Map::putAll); } /** * Distinguishes groups of fields to be aggregated. */ public enum Group { /** * Identifies fields that were declared directly by the class in question. */ DECLARED(c -> Stream.of(c.getDeclaredFields())), /** * Identifies fields that are declared public. */ PUBLIC(c -> Stream.of(c.getFields())); private final Function, Stream> streaming; Group(Function, Stream> streaming) { this.streaming = streaming; } } public static class AccessException extends IllegalStateException { private AccessException(final Throwable cause) { super(cause.getMessage(), cause); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy