FieldUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.reflect;

  18. import java.lang.annotation.Annotation;
  19. import java.lang.reflect.Field;
  20. import java.lang.reflect.Modifier;
  21. import java.util.ArrayList;
  22. import java.util.Collections;
  23. import java.util.List;
  24. import java.util.Objects;
  25. import java.util.stream.Collectors;

  26. import org.apache.commons.lang3.ArrayUtils;
  27. import org.apache.commons.lang3.ClassUtils;
  28. import org.apache.commons.lang3.JavaVersion;
  29. import org.apache.commons.lang3.StringUtils;
  30. import org.apache.commons.lang3.SystemUtils;
  31. import org.apache.commons.lang3.Validate;

  32. /**
  33.  * Utilities for working with {@link Field}s by reflection. Adapted and refactored from the dormant [reflect] Commons
  34.  * sandbox component.
  35.  * <p>
  36.  * The ability is provided to break the scoping restrictions coded by the programmer. This can allow fields to be
  37.  * changed that shouldn't be. This facility should be used with care.
  38.  * </p>
  39.  * @since 2.5
  40.  */
  41. public class FieldUtils {

  42.     /**
  43.      * Gets all fields of the given class and its parents (if any).
  44.      *
  45.      * @param cls
  46.      *            the {@link Class} to query
  47.      * @return an array of Fields (possibly empty).
  48.      * @throws NullPointerException
  49.      *             if the class is {@code null}
  50.      * @since 3.2
  51.      */
  52.     public static Field[] getAllFields(final Class<?> cls) {
  53.         return getAllFieldsList(cls).toArray(ArrayUtils.EMPTY_FIELD_ARRAY);
  54.     }

  55.     /**
  56.      * Gets all fields of the given class and its parents (if any).
  57.      *
  58.      * @param cls
  59.      *            the {@link Class} to query
  60.      * @return a list of Fields (possibly empty).
  61.      * @throws NullPointerException
  62.      *             if the class is {@code null}
  63.      * @since 3.2
  64.      */
  65.     public static List<Field> getAllFieldsList(final Class<?> cls) {
  66.         Objects.requireNonNull(cls, "cls");
  67.         final List<Field> allFields = new ArrayList<>();
  68.         Class<?> currentClass = cls;
  69.         while (currentClass != null) {
  70.             final Field[] declaredFields = currentClass.getDeclaredFields();
  71.             Collections.addAll(allFields, declaredFields);
  72.             currentClass = currentClass.getSuperclass();
  73.         }
  74.         return allFields;
  75.     }

  76.     /**
  77.      * Gets an accessible {@link Field} by name respecting scope. Only the specified class will be considered.
  78.      *
  79.      * @param cls
  80.      *            the {@link Class} to reflect, must not be {@code null}
  81.      * @param fieldName
  82.      *            the field name to obtain
  83.      * @return the Field object
  84.      * @throws NullPointerException
  85.      *             if the class is {@code null}
  86.      * @throws IllegalArgumentException
  87.      *             if the field name is {@code null}, blank, or empty
  88.      */
  89.     public static Field getDeclaredField(final Class<?> cls, final String fieldName) {
  90.         return getDeclaredField(cls, fieldName, false);
  91.     }

  92.     /**
  93.      * Gets an accessible {@link Field} by name, breaking scope if requested. Only the specified class will be
  94.      * considered.
  95.      *
  96.      * @param cls
  97.      *            the {@link Class} to reflect, must not be {@code null}
  98.      * @param fieldName
  99.      *            the field name to obtain
  100.      * @param forceAccess
  101.      *            whether to break scope restrictions using the
  102.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  103.      *            match {@code public} fields.
  104.      * @return the Field object
  105.      * @throws NullPointerException
  106.      *             if the class is {@code null}
  107.      * @throws IllegalArgumentException
  108.      *             if the field name is {@code null}, blank, or empty
  109.      */
  110.     public static Field getDeclaredField(final Class<?> cls, final String fieldName, final boolean forceAccess) {
  111.         Objects.requireNonNull(cls, "cls");
  112.         Validate.isTrue(StringUtils.isNotBlank(fieldName), "The field name must not be blank/empty");
  113.         try {
  114.             // only consider the specified class by using getDeclaredField()
  115.             final Field field = cls.getDeclaredField(fieldName);
  116.             if (!MemberUtils.isAccessible(field)) {
  117.                 if (!forceAccess) {
  118.                     return null;
  119.                 }
  120.                 field.setAccessible(true);
  121.             }
  122.             return field;
  123.         } catch (final NoSuchFieldException ignored) {
  124.             // ignore
  125.         }
  126.         return null;
  127.     }

  128.     /**
  129.      * Gets an accessible {@link Field} by name respecting scope. Superclasses/interfaces will be considered.
  130.      *
  131.      * @param cls
  132.      *            the {@link Class} to reflect, must not be {@code null}
  133.      * @param fieldName
  134.      *            the field name to obtain
  135.      * @return the Field object
  136.      * @throws NullPointerException
  137.      *             if the class is {@code null}
  138.      * @throws IllegalArgumentException
  139.      *             if the field name is {@code null}, blank, or empty
  140.      */
  141.     public static Field getField(final Class<?> cls, final String fieldName) {
  142.         return MemberUtils.setAccessibleWorkaround(getField(cls, fieldName, false));
  143.     }

  144.     /**
  145.      * Gets an accessible {@link Field} by name, breaking scope if requested. Superclasses/interfaces will be
  146.      * considered.
  147.      *
  148.      * @param cls
  149.      *            the {@link Class} to reflect, must not be {@code null}
  150.      * @param fieldName
  151.      *            the field name to obtain
  152.      * @param forceAccess
  153.      *            whether to break scope restrictions using the
  154.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  155.      *            match {@code public} fields.
  156.      * @return the Field object
  157.      * @throws NullPointerException if the class is {@code null}
  158.      * @throws IllegalArgumentException if the field name is blank or empty or is matched at multiple places
  159.      * in the inheritance hierarchy
  160.      */
  161.     public static Field getField(final Class<?> cls, final String fieldName, final boolean forceAccess) {
  162.         Objects.requireNonNull(cls, "cls");
  163.         Validate.isTrue(StringUtils.isNotBlank(fieldName), "The field name must not be blank/empty");
  164.         // FIXME is this workaround still needed? lang requires Java 6
  165.         // Sun Java 1.3 has a bugged implementation of getField hence we write the
  166.         // code ourselves

  167.         // getField() will return the Field object with the declaring class
  168.         // set correctly to the class that declares the field. Thus requesting the
  169.         // field on a subclass will return the field from the superclass.
  170.         //
  171.         // priority order for lookup:
  172.         // searchclass private/protected/package/public
  173.         // superclass protected/package/public
  174.         // private/different package blocks access to further superclasses
  175.         // implementedinterface public

  176.         // check up the superclass hierarchy
  177.         for (Class<?> acls = cls; acls != null; acls = acls.getSuperclass()) {
  178.             try {
  179.                 final Field field = acls.getDeclaredField(fieldName);
  180.                 // getDeclaredField checks for non-public scopes as well
  181.                 // and it returns accurate results
  182.                 if (!MemberUtils.isPublic(field)) {
  183.                     if (!forceAccess) {
  184.                         continue;
  185.                     }
  186.                     field.setAccessible(true);
  187.                 }
  188.                 return field;
  189.             } catch (final NoSuchFieldException ignored) {
  190.                 // ignore
  191.             }
  192.         }
  193.         // check the public interface case. This must be manually searched for
  194.         // incase there is a public supersuperclass field hidden by a private/package
  195.         // superclass field.
  196.         Field match = null;
  197.         for (final Class<?> class1 : ClassUtils.getAllInterfaces(cls)) {
  198.             try {
  199.                 final Field test = class1.getField(fieldName);
  200.                 Validate.isTrue(match == null, "Reference to field %s is ambiguous relative to %s"
  201.                         + "; a matching field exists on two or more implemented interfaces.", fieldName, cls);
  202.                 match = test;
  203.             } catch (final NoSuchFieldException ignored) {
  204.                 // ignore
  205.             }
  206.         }
  207.         return match;
  208.     }

  209.     /**
  210.      * Gets all fields of the given class and its parents (if any) that are annotated with the given annotation.
  211.      * @param cls
  212.      *            the {@link Class} to query
  213.      * @param annotationCls
  214.      *            the {@link Annotation} that must be present on a field to be matched
  215.      * @return a list of Fields (possibly empty).
  216.      * @throws NullPointerException
  217.      *            if the class or annotation are {@code null}
  218.      * @since 3.4
  219.      */
  220.     public static List<Field> getFieldsListWithAnnotation(final Class<?> cls, final Class<? extends Annotation> annotationCls) {
  221.         Objects.requireNonNull(annotationCls, "annotationCls");
  222.         return getAllFieldsList(cls).stream().filter(field -> field.getAnnotation(annotationCls) != null).collect(Collectors.toList());
  223.     }

  224.     /**
  225.      * Gets all fields of the given class and its parents (if any) that are annotated with the given annotation.
  226.      * @param cls
  227.      *            the {@link Class} to query
  228.      * @param annotationCls
  229.      *            the {@link Annotation} that must be present on a field to be matched
  230.      * @return an array of Fields (possibly empty).
  231.      * @throws NullPointerException
  232.      *            if the class or annotation are {@code null}
  233.      * @since 3.4
  234.      */
  235.     public static Field[] getFieldsWithAnnotation(final Class<?> cls, final Class<? extends Annotation> annotationCls) {
  236.         return getFieldsListWithAnnotation(cls, annotationCls).toArray(ArrayUtils.EMPTY_FIELD_ARRAY);
  237.     }

  238.     /**
  239.      * Reads the named {@code public} {@link Field}. Only the class of the specified object will be considered.
  240.      *
  241.      * @param target
  242.      *            the object to reflect, must not be {@code null}
  243.      * @param fieldName
  244.      *            the field name to obtain
  245.      * @return the value of the field
  246.      * @throws NullPointerException
  247.      *             if {@code target} is {@code null}
  248.      * @throws IllegalArgumentException
  249.      *             if {@code fieldName} is {@code null}, blank or empty, or could not be found
  250.      * @throws IllegalAccessException
  251.      *             if the named field is not {@code public}
  252.      */
  253.     public static Object readDeclaredField(final Object target, final String fieldName) throws IllegalAccessException {
  254.         return readDeclaredField(target, fieldName, false);
  255.     }

  256.     /**
  257.      * Gets a {@link Field} value by name. Only the class of the specified object will be considered.
  258.      *
  259.      * @param target
  260.      *            the object to reflect, must not be {@code null}
  261.      * @param fieldName
  262.      *            the field name to obtain
  263.      * @param forceAccess
  264.      *            whether to break scope restrictions using the
  265.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  266.      *            match public fields.
  267.      * @return the Field object
  268.      * @throws NullPointerException
  269.      *             if {@code target} is {@code null}
  270.      * @throws IllegalArgumentException
  271.      *             if {@code fieldName} is {@code null}, blank or empty, or could not be found
  272.      * @throws IllegalAccessException
  273.      *             if the field is not made accessible
  274.      */
  275.     public static Object readDeclaredField(final Object target, final String fieldName, final boolean forceAccess) throws IllegalAccessException {
  276.         Objects.requireNonNull(target, "target");
  277.         final Class<?> cls = target.getClass();
  278.         final Field field = getDeclaredField(cls, fieldName, forceAccess);
  279.         Validate.isTrue(field != null, "Cannot locate declared field %s.%s", cls, fieldName);
  280.         // already forced access above, don't repeat it here:
  281.         return readField(field, target, false);
  282.     }

  283.     /**
  284.      * Gets the value of a {@code static} {@link Field} by name. The field must be {@code public}. Only the specified
  285.      * class will be considered.
  286.      *
  287.      * @param cls
  288.      *            the {@link Class} to reflect, must not be {@code null}
  289.      * @param fieldName
  290.      *            the field name to obtain
  291.      * @return the value of the field
  292.      * @throws NullPointerException
  293.      *             if the class is {@code null}, or the field could not be found
  294.      * @throws IllegalArgumentException
  295.      *             if the field name is {@code null}, blank, empty, or is not {@code static}
  296.      * @throws IllegalAccessException
  297.      *             if the field is not accessible
  298.      */
  299.     public static Object readDeclaredStaticField(final Class<?> cls, final String fieldName) throws IllegalAccessException {
  300.         return readDeclaredStaticField(cls, fieldName, false);
  301.     }

  302.     /**
  303.      * Gets the value of a {@code static} {@link Field} by name. Only the specified class will be considered.
  304.      *
  305.      * @param cls
  306.      *            the {@link Class} to reflect, must not be {@code null}
  307.      * @param fieldName
  308.      *            the field name to obtain
  309.      * @param forceAccess
  310.      *            whether to break scope restrictions using the
  311.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  312.      *            match {@code public} fields.
  313.      * @return the Field object
  314.      * @throws NullPointerException
  315.      *             if the class is {@code null}, or the field could not be found
  316.      * @throws IllegalArgumentException
  317.      *             if the field name is blank or empty, is not {@code static}
  318.      * @throws IllegalAccessException
  319.      *             if the field is not made accessible
  320.      */
  321.     public static Object readDeclaredStaticField(final Class<?> cls, final String fieldName, final boolean forceAccess) throws IllegalAccessException {
  322.         final Field field = getDeclaredField(cls, fieldName, forceAccess);
  323.         Validate.notNull(field, "Cannot locate declared field %s.%s", cls.getName(), fieldName);
  324.         // already forced access above, don't repeat it here:
  325.         return readStaticField(field, false);
  326.     }

  327.     /**
  328.      * Reads an accessible {@link Field}.
  329.      *
  330.      * @param field
  331.      *            the field to use
  332.      * @param target
  333.      *            the object to call on, may be {@code null} for {@code static} fields
  334.      * @return the field value
  335.      * @throws NullPointerException
  336.      *             if the field is {@code null}
  337.      * @throws IllegalAccessException
  338.      *             if the field is not accessible
  339.      */
  340.     public static Object readField(final Field field, final Object target) throws IllegalAccessException {
  341.         return readField(field, target, false);
  342.     }

  343.     /**
  344.      * Reads a {@link Field}.
  345.      *
  346.      * @param field
  347.      *            the field to use
  348.      * @param target
  349.      *            the object to call on, may be {@code null} for {@code static} fields
  350.      * @param forceAccess
  351.      *            whether to break scope restrictions using the
  352.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method.
  353.      * @return the field value
  354.      * @throws NullPointerException
  355.      *             if the field is {@code null}
  356.      * @throws IllegalAccessException
  357.      *             if the field is not made accessible
  358.      */
  359.     public static Object readField(final Field field, final Object target, final boolean forceAccess) throws IllegalAccessException {
  360.         Objects.requireNonNull(field, "field");
  361.         if (forceAccess && !field.isAccessible()) {
  362.             field.setAccessible(true);
  363.         } else {
  364.             MemberUtils.setAccessibleWorkaround(field);
  365.         }
  366.         return field.get(target);
  367.     }

  368.     /**
  369.      * Reads the named {@code public} {@link Field}. Superclasses will be considered.
  370.      *
  371.      * @param target
  372.      *            the object to reflect, must not be {@code null}
  373.      * @param fieldName
  374.      *            the field name to obtain
  375.      * @return the value of the field
  376.      * @throws NullPointerException
  377.      *             if the target is {@code null}
  378.      * @throws IllegalArgumentException
  379.      *             if the field name is {@code null}, blank, empty, or could not be found
  380.      * @throws IllegalAccessException
  381.      *             if the named field is not {@code public}
  382.      */
  383.     public static Object readField(final Object target, final String fieldName) throws IllegalAccessException {
  384.         return readField(target, fieldName, false);
  385.     }

  386.     /**
  387.      * Reads the named {@link Field}. Superclasses will be considered.
  388.      *
  389.      * @param target
  390.      *            the object to reflect, must not be {@code null}
  391.      * @param fieldName
  392.      *            the field name to obtain
  393.      * @param forceAccess
  394.      *            whether to break scope restrictions using the
  395.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  396.      *            match {@code public} fields.
  397.      * @return the field value
  398.      * @throws NullPointerException
  399.      *             if {@code target} is {@code null}
  400.      * @throws IllegalArgumentException
  401.      *             if the field name is {@code null}, blank, empty, or could not be found
  402.      * @throws IllegalAccessException
  403.      *             if the named field is not made accessible
  404.      */
  405.     public static Object readField(final Object target, final String fieldName, final boolean forceAccess) throws IllegalAccessException {
  406.         Objects.requireNonNull(target, "target");
  407.         final Class<?> cls = target.getClass();
  408.         final Field field = getField(cls, fieldName, forceAccess);
  409.         Validate.isTrue(field != null, "Cannot locate field %s on %s", fieldName, cls);
  410.         // already forced access above, don't repeat it here:
  411.         return readField(field, target, false);
  412.     }

  413.     /**
  414.      * Reads the named {@code public static} {@link Field}. Superclasses will be considered.
  415.      *
  416.      * @param cls
  417.      *            the {@link Class} to reflect, must not be {@code null}
  418.      * @param fieldName
  419.      *            the field name to obtain
  420.      * @return the value of the field
  421.      * @throws NullPointerException
  422.      *             if the class is {@code null}, or the field could not be found
  423.      * @throws IllegalArgumentException
  424.      *             if the field name is {@code null}, blank or empty, or is not {@code static}
  425.      * @throws IllegalAccessException
  426.      *             if the field is not accessible
  427.      */
  428.     public static Object readStaticField(final Class<?> cls, final String fieldName) throws IllegalAccessException {
  429.         return readStaticField(cls, fieldName, false);
  430.     }

  431.     /**
  432.      * Reads the named {@code static} {@link Field}. Superclasses will be considered.
  433.      *
  434.      * @param cls
  435.      *            the {@link Class} to reflect, must not be {@code null}
  436.      * @param fieldName
  437.      *            the field name to obtain
  438.      * @param forceAccess
  439.      *            whether to break scope restrictions using the
  440.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  441.      *            match {@code public} fields.
  442.      * @return the Field object
  443.      * @throws NullPointerException
  444.      *             if the class is {@code null}, or the field could not be found
  445.      * @throws IllegalArgumentException
  446.      *             if the field name is {@code null}, blank or empty, or is not {@code static}
  447.      * @throws IllegalAccessException
  448.      *             if the field is not made accessible
  449.      */
  450.     public static Object readStaticField(final Class<?> cls, final String fieldName, final boolean forceAccess) throws IllegalAccessException {
  451.         final Field field = getField(cls, fieldName, forceAccess);
  452.         Validate.notNull(field, "Cannot locate field '%s' on %s", fieldName, cls);
  453.         // already forced access above, don't repeat it here:
  454.         return readStaticField(field, false);
  455.     }

  456.     /**
  457.      * Reads an accessible {@code static} {@link Field}.
  458.      *
  459.      * @param field
  460.      *            to read
  461.      * @return the field value
  462.      * @throws NullPointerException
  463.      *             if the field is {@code null}
  464.      * @throws IllegalArgumentException
  465.      *             if the field is not {@code static}
  466.      * @throws IllegalAccessException
  467.      *             if the field is not accessible
  468.      */
  469.     public static Object readStaticField(final Field field) throws IllegalAccessException {
  470.         return readStaticField(field, false);
  471.     }

  472.     /**
  473.      * Reads a static {@link Field}.
  474.      *
  475.      * @param field
  476.      *            to read
  477.      * @param forceAccess
  478.      *            whether to break scope restrictions using the
  479.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method.
  480.      * @return the field value
  481.      * @throws NullPointerException
  482.      *             if the field is {@code null}
  483.      * @throws IllegalArgumentException
  484.      *             if the field is not {@code static}
  485.      * @throws IllegalAccessException
  486.      *             if the field is not made accessible
  487.      */
  488.     public static Object readStaticField(final Field field, final boolean forceAccess) throws IllegalAccessException {
  489.         Objects.requireNonNull(field, "field");
  490.         Validate.isTrue(MemberUtils.isStatic(field), "The field '%s' is not static", field.getName());
  491.         return readField(field, (Object) null, forceAccess);
  492.     }

  493.     /**
  494.      * Removes the final modifier from a {@link Field}.
  495.      *
  496.      * @param field
  497.      *            to remove the final modifier
  498.      * @throws NullPointerException
  499.      *             if the field is {@code null}
  500.      * @since 3.2
  501.      */
  502.     public static void removeFinalModifier(final Field field) {
  503.         removeFinalModifier(field, true);
  504.     }

  505.     /**
  506.      * Removes the final modifier from a {@link Field}.
  507.      *
  508.      * @param field
  509.      *            to remove the final modifier
  510.      * @param forceAccess
  511.      *            whether to break scope restrictions using the
  512.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  513.      *            match {@code public} fields.
  514.      * @throws NullPointerException
  515.      *             if the field is {@code null}
  516.      * @deprecated As of Java 12, we can no longer drop the {@code final} modifier, thus
  517.      *             rendering this method obsolete. The JDK discussion about this change can be found
  518.      *             here: https://mail.openjdk.java.net/pipermail/core-libs-dev/2018-November/056486.html
  519.      * @since 3.3
  520.      */
  521.     @Deprecated
  522.     public static void removeFinalModifier(final Field field, final boolean forceAccess) {
  523.         Objects.requireNonNull(field, "field");

  524.         try {
  525.             if (Modifier.isFinal(field.getModifiers())) {
  526.                 // Do all JREs implement Field with a private ivar called "modifiers"?
  527.                 final Field modifiersField = Field.class.getDeclaredField("modifiers");
  528.                 final boolean doForceAccess = forceAccess && !modifiersField.isAccessible();
  529.                 if (doForceAccess) {
  530.                     modifiersField.setAccessible(true);
  531.                 }
  532.                 try {
  533.                     modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
  534.                 } finally {
  535.                     if (doForceAccess) {
  536.                         modifiersField.setAccessible(false);
  537.                     }
  538.                 }
  539.             }
  540.         } catch (final NoSuchFieldException | IllegalAccessException e) {
  541.             if (SystemUtils.isJavaVersionAtLeast(JavaVersion.JAVA_12)) {
  542.               throw new UnsupportedOperationException(
  543.                   "In java 12+ final cannot be removed.",
  544.                   e
  545.               );
  546.             }
  547.             // else no exception is thrown because we can modify final.
  548.         }
  549.     }

  550.     /**
  551.      * Writes a {@code public} {@link Field}. Only the specified class will be considered.
  552.      *
  553.      * @param target
  554.      *            the object to reflect, must not be {@code null}
  555.      * @param fieldName
  556.      *            the field name to obtain
  557.      * @param value
  558.      *            to set
  559.      * @throws NullPointerException
  560.      *             if {@code target} is {@code null}
  561.      * @throws IllegalArgumentException
  562.      *             if {@code fieldName} is {@code null}, blank or empty, or could not be found,
  563.      *             or {@code value} is not assignable
  564.      * @throws IllegalAccessException
  565.      *             if the field is not made accessible
  566.      */
  567.     public static void writeDeclaredField(final Object target, final String fieldName, final Object value) throws IllegalAccessException {
  568.         writeDeclaredField(target, fieldName, value, false);
  569.     }

  570.     /**
  571.      * Writes a {@code public} {@link Field}. Only the specified class will be considered.
  572.      *
  573.      * @param target
  574.      *            the object to reflect, must not be {@code null}
  575.      * @param fieldName
  576.      *            the field name to obtain
  577.      * @param value
  578.      *            to set
  579.      * @param forceAccess
  580.      *            whether to break scope restrictions using the
  581.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  582.      *            match {@code public} fields.
  583.      * @throws IllegalArgumentException
  584.      *             if {@code fieldName} is {@code null}, blank or empty, or could not be found,
  585.      *             or {@code value} is not assignable
  586.      * @throws IllegalAccessException
  587.      *             if the field is not made accessible
  588.      */
  589.     public static void writeDeclaredField(final Object target, final String fieldName, final Object value, final boolean forceAccess)
  590.             throws IllegalAccessException {
  591.         Objects.requireNonNull(target, "target");
  592.         final Class<?> cls = target.getClass();
  593.         final Field field = getDeclaredField(cls, fieldName, forceAccess);
  594.         Validate.isTrue(field != null, "Cannot locate declared field %s.%s", cls.getName(), fieldName);
  595.         // already forced access above, don't repeat it here:
  596.         writeField(field, target, value, false);
  597.     }

  598.     /**
  599.      * Writes a named {@code public static} {@link Field}. Only the specified class will be considered.
  600.      *
  601.      * @param cls
  602.      *            {@link Class} on which the field is to be found
  603.      * @param fieldName
  604.      *            to write
  605.      * @param value
  606.      *            to set
  607.      * @throws NullPointerException
  608.      *             if {@code cls} is {@code null} or the field cannot be located
  609.      * @throws IllegalArgumentException
  610.      *             if the field name is {@code null}, blank, empty, not {@code static}, or {@code value} is not assignable
  611.      * @throws IllegalAccessException
  612.      *             if the field is not {@code public} or is {@code final}
  613.      */
  614.     public static void writeDeclaredStaticField(final Class<?> cls, final String fieldName, final Object value) throws IllegalAccessException {
  615.         writeDeclaredStaticField(cls, fieldName, value, false);
  616.     }

  617.     /**
  618.      * Writes a named {@code static} {@link Field}. Only the specified class will be considered.
  619.      *
  620.      * @param cls
  621.      *            {@link Class} on which the field is to be found
  622.      * @param fieldName
  623.      *            to write
  624.      * @param value
  625.      *            to set
  626.      * @param forceAccess
  627.      *            whether to break scope restrictions using the {@code AccessibleObject#setAccessible(boolean)} method.
  628.      *            {@code false} will only match {@code public} fields.
  629.      * @throws NullPointerException
  630.      *             if {@code cls} is {@code null} or the field cannot be located
  631.      * @throws IllegalArgumentException
  632.      *             if the field name is {@code null}, blank, empty, not {@code static}, or {@code value} is not assignable
  633.      * @throws IllegalAccessException
  634.      *             if the field is not made accessible or is {@code final}
  635.      */
  636.     public static void writeDeclaredStaticField(final Class<?> cls, final String fieldName, final Object value, final boolean forceAccess)
  637.             throws IllegalAccessException {
  638.         final Field field = getDeclaredField(cls, fieldName, forceAccess);
  639.         Validate.notNull(field, "Cannot locate declared field %s.%s", cls.getName(), fieldName);
  640.         // already forced access above, don't repeat it here:
  641.         writeField(field, (Object) null, value, false);
  642.     }

  643.     /**
  644.      * Writes an accessible {@link Field}.
  645.      *
  646.      * @param field
  647.      *            to write
  648.      * @param target
  649.      *            the object to call on, may be {@code null} for {@code static} fields
  650.      * @param value
  651.      *            to set
  652.      * @throws NullPointerException
  653.      *             if the field is {@code null}
  654.      * @throws IllegalArgumentException
  655.      *             if {@code value} is not assignable
  656.      * @throws IllegalAccessException
  657.      *             if the field is not accessible or is {@code final}
  658.      */
  659.     public static void writeField(final Field field, final Object target, final Object value) throws IllegalAccessException {
  660.         writeField(field, target, value, false);
  661.     }

  662.     /**
  663.      * Writes a {@link Field}.
  664.      *
  665.      * @param field
  666.      *            to write
  667.      * @param target
  668.      *            the object to call on, may be {@code null} for {@code static} fields
  669.      * @param value
  670.      *            to set
  671.      * @param forceAccess
  672.      *            whether to break scope restrictions using the
  673.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  674.      *            match {@code public} fields.
  675.      * @throws NullPointerException
  676.      *             if the field is {@code null}
  677.      * @throws IllegalArgumentException
  678.      *             if {@code value} is not assignable
  679.      * @throws IllegalAccessException
  680.      *             if the field is not made accessible or is {@code final}
  681.      */
  682.     public static void writeField(final Field field, final Object target, final Object value, final boolean forceAccess)
  683.             throws IllegalAccessException {
  684.         Objects.requireNonNull(field, "field");
  685.         if (forceAccess && !field.isAccessible()) {
  686.             field.setAccessible(true);
  687.         } else {
  688.             MemberUtils.setAccessibleWorkaround(field);
  689.         }
  690.         field.set(target, value);
  691.     }

  692.     /**
  693.      * Writes a {@code public} {@link Field}. Superclasses will be considered.
  694.      *
  695.      * @param target
  696.      *            the object to reflect, must not be {@code null}
  697.      * @param fieldName
  698.      *            the field name to obtain
  699.      * @param value
  700.      *            to set
  701.      * @throws NullPointerException
  702.      *             if {@code target} is {@code null}
  703.      * @throws IllegalArgumentException
  704.      *             if {@code fieldName} is {@code null}, blank, empty, or could not be found,
  705.      *             or {@code value} is not assignable
  706.      * @throws IllegalAccessException
  707.      *             if the field is not accessible
  708.      */
  709.     public static void writeField(final Object target, final String fieldName, final Object value) throws IllegalAccessException {
  710.         writeField(target, fieldName, value, false);
  711.     }

  712.     /**
  713.      * Writes a {@link Field}. Superclasses will be considered.
  714.      *
  715.      * @param target
  716.      *            the object to reflect, must not be {@code null}
  717.      * @param fieldName
  718.      *            the field name to obtain
  719.      * @param value
  720.      *            to set
  721.      * @param forceAccess
  722.      *            whether to break scope restrictions using the
  723.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  724.      *            match {@code public} fields.
  725.      * @throws NullPointerException
  726.      *             if {@code target} is {@code null}
  727.      * @throws IllegalArgumentException
  728.      *             if {@code fieldName} is {@code null}, blank, empty, or could not be found,
  729.      *             or {@code value} is not assignable
  730.      * @throws IllegalAccessException
  731.      *             if the field is not made accessible
  732.      */
  733.     public static void writeField(final Object target, final String fieldName, final Object value, final boolean forceAccess)
  734.             throws IllegalAccessException {
  735.         Objects.requireNonNull(target, "target");
  736.         final Class<?> cls = target.getClass();
  737.         final Field field = getField(cls, fieldName, forceAccess);
  738.         Validate.isTrue(field != null, "Cannot locate declared field %s.%s", cls.getName(), fieldName);
  739.         // already forced access above, don't repeat it here:
  740.         writeField(field, target, value, false);
  741.     }

  742.     /**
  743.      * Writes a named {@code public static} {@link Field}. Superclasses will be considered.
  744.      *
  745.      * @param cls
  746.      *            {@link Class} on which the field is to be found
  747.      * @param fieldName
  748.      *            to write
  749.      * @param value
  750.      *            to set
  751.      * @throws NullPointerException
  752.      *             if {@code target} is {@code null}
  753.      * @throws IllegalArgumentException
  754.      *             if {@code fieldName} is {@code null}, blank or empty, the field cannot be located or is
  755.      *             not {@code static}, or {@code value} is not assignable
  756.      * @throws IllegalAccessException
  757.      *             if the field is not {@code public} or is {@code final}
  758.      */
  759.     public static void writeStaticField(final Class<?> cls, final String fieldName, final Object value) throws IllegalAccessException {
  760.         writeStaticField(cls, fieldName, value, false);
  761.     }

  762.     /**
  763.      * Writes a named {@code static} {@link Field}. Superclasses will be considered.
  764.      *
  765.      * @param cls
  766.      *            {@link Class} on which the field is to be found
  767.      * @param fieldName
  768.      *            to write
  769.      * @param value
  770.      *            to set
  771.      * @param forceAccess
  772.      *            whether to break scope restrictions using the
  773.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  774.      *            match {@code public} fields.
  775.      * @throws NullPointerException
  776.      *             if {@code cls} is {@code null} or the field cannot be located
  777.      * @throws IllegalArgumentException
  778.      *             if {@code fieldName} is {@code null}, blank or empty, the field not {@code static}, or {@code value} is not assignable
  779.      * @throws IllegalAccessException
  780.      *             if the field is not made accessible or is {@code final}
  781.      */
  782.     public static void writeStaticField(final Class<?> cls, final String fieldName, final Object value, final boolean forceAccess)
  783.             throws IllegalAccessException {
  784.         final Field field = getField(cls, fieldName, forceAccess);
  785.         Validate.notNull(field, "Cannot locate field %s on %s", fieldName, cls);
  786.         // already forced access above, don't repeat it here:
  787.         writeStaticField(field, value, false);
  788.     }

  789.     /**
  790.      * Writes a {@code public static} {@link Field}.
  791.      *
  792.      * @param field
  793.      *            to write
  794.      * @param value
  795.      *            to set
  796.      * @throws NullPointerException
  797.      *              if the field is {@code null}
  798.      * @throws IllegalArgumentException
  799.      *              if the field is not {@code static}, or {@code value} is not assignable
  800.      * @throws IllegalAccessException
  801.      *             if the field is not {@code public} or is {@code final}
  802.      */
  803.     public static void writeStaticField(final Field field, final Object value) throws IllegalAccessException {
  804.         writeStaticField(field, value, false);
  805.     }

  806.     /**
  807.      * Writes a static {@link Field}.
  808.      *
  809.      * @param field
  810.      *            to write
  811.      * @param value
  812.      *            to set
  813.      * @param forceAccess
  814.      *            whether to break scope restrictions using the
  815.      *            {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} method. {@code false} will only
  816.      *            match {@code public} fields.
  817.      * @throws NullPointerException
  818.      *              if the field is {@code null}
  819.      * @throws IllegalArgumentException
  820.      *              if the field is not {@code static}, or {@code value} is not assignable
  821.      * @throws IllegalAccessException
  822.      *             if the field is not made accessible or is {@code final}
  823.      */
  824.     public static void writeStaticField(final Field field, final Object value, final boolean forceAccess) throws IllegalAccessException {
  825.         Objects.requireNonNull(field, "field");
  826.         Validate.isTrue(MemberUtils.isStatic(field), "The field %s.%s is not static", field.getDeclaringClass().getName(),
  827.                 field.getName());
  828.         writeField(field, (Object) null, value, forceAccess);
  829.     }

  830.     /**
  831.      * {@link FieldUtils} instances should NOT be constructed in standard programming.
  832.      * <p>
  833.      * This constructor is {@code public} to permit tools that require a JavaBean instance to operate.
  834.      * </p>
  835.      *
  836.      * @deprecated TODO Make private in 4.0.
  837.      */
  838.     @Deprecated
  839.     public FieldUtils() {
  840.         // empty
  841.     }
  842. }