ConstructorUtils.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.reflect.Constructor;
  19. import java.lang.reflect.InvocationTargetException;
  20. import java.util.Objects;

  21. import org.apache.commons.lang3.ArrayUtils;
  22. import org.apache.commons.lang3.ClassUtils;

  23. /**
  24.  *  Utility reflection methods focused on constructors, modeled after
  25.  * {@link MethodUtils}.
  26.  *
  27.  * <h2>Known Limitations</h2>
  28.  * <h3>Accessing Public Constructors In A Default Access Superclass</h3>
  29.  * <p>There is an issue when invoking {@code public} constructors
  30.  * contained in a default access superclass. Reflection correctly locates these
  31.  * constructors and assigns them as {@code public}. However, an
  32.  * {@link IllegalAccessException} is thrown if the constructor is
  33.  * invoked.</p>
  34.  *
  35.  * <p>{@link ConstructorUtils} contains a workaround for this situation: it
  36.  * will attempt to call {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} on this constructor. If this
  37.  * call succeeds, then the method can be invoked as normal. This call will only
  38.  * succeed when the application has sufficient security privileges. If this call
  39.  * fails then a warning will be logged and the method may fail.</p>
  40.  *
  41.  * @since 2.5
  42.  */
  43. public class ConstructorUtils {

  44.     /**
  45.      * Finds a constructor given a class and signature, checking accessibility.
  46.      *
  47.      * <p>This finds the constructor and ensures that it is accessible.
  48.      * The constructor signature must match the parameter types exactly.</p>
  49.      *
  50.      * @param <T> the constructor type
  51.      * @param cls the class to find a constructor for, not {@code null}
  52.      * @param parameterTypes the array of parameter types, {@code null} treated as empty
  53.      * @return the constructor, {@code null} if no matching accessible constructor found
  54.      * @see Class#getConstructor
  55.      * @see #getAccessibleConstructor(java.lang.reflect.Constructor)
  56.      * @throws NullPointerException if {@code cls} is {@code null}
  57.      */
  58.     public static <T> Constructor<T> getAccessibleConstructor(final Class<T> cls,
  59.             final Class<?>... parameterTypes) {
  60.         Objects.requireNonNull(cls, "cls");
  61.         try {
  62.             return getAccessibleConstructor(cls.getConstructor(parameterTypes));
  63.         } catch (final NoSuchMethodException e) {
  64.             return null;
  65.         }
  66.     }

  67.     /**
  68.      * Checks if the specified constructor is accessible.
  69.      *
  70.      * <p>This simply ensures that the constructor is accessible.</p>
  71.      *
  72.      * @param <T> the constructor type
  73.      * @param ctor  the prototype constructor object, not {@code null}
  74.      * @return the constructor, {@code null} if no matching accessible constructor found
  75.      * @see SecurityManager
  76.      * @throws NullPointerException if {@code ctor} is {@code null}
  77.      */
  78.     public static <T> Constructor<T> getAccessibleConstructor(final Constructor<T> ctor) {
  79.         Objects.requireNonNull(ctor, "ctor");
  80.         return MemberUtils.isAccessible(ctor)
  81.                 && isAccessible(ctor.getDeclaringClass()) ? ctor : null;
  82.     }

  83.     /**
  84.      * Finds an accessible constructor with compatible parameters.
  85.      *
  86.      * <p>This checks all the constructor and finds one with compatible parameters
  87.      * This requires that every parameter is assignable from the given parameter types.
  88.      * This is a more flexible search than the normal exact matching algorithm.</p>
  89.      *
  90.      * <p>First it checks if there is a constructor matching the exact signature.
  91.      * If not then all the constructors of the class are checked to see if their
  92.      * signatures are assignment-compatible with the parameter types.
  93.      * The first assignment-compatible matching constructor is returned.</p>
  94.      *
  95.      * @param <T> the constructor type
  96.      * @param cls  the class to find a constructor for, not {@code null}
  97.      * @param parameterTypes find method with compatible parameters
  98.      * @return the constructor, null if no matching accessible constructor found
  99.      * @throws NullPointerException if {@code cls} is {@code null}
  100.      */
  101.     public static <T> Constructor<T> getMatchingAccessibleConstructor(final Class<T> cls,
  102.             final Class<?>... parameterTypes) {
  103.         Objects.requireNonNull(cls, "cls");
  104.         // see if we can find the constructor directly
  105.         // most of the time this works and it's much faster
  106.         try {
  107.             return MemberUtils.setAccessibleWorkaround(cls.getConstructor(parameterTypes));
  108.         } catch (final NoSuchMethodException ignored) {
  109.             // ignore
  110.         }
  111.         Constructor<T> result = null;
  112.         /*
  113.          * (1) Class.getConstructors() is documented to return Constructor<T> so as
  114.          * long as the array is not subsequently modified, everything's fine.
  115.          */
  116.         final Constructor<?>[] ctors = cls.getConstructors();

  117.         // return best match:
  118.         for (Constructor<?> ctor : ctors) {
  119.             // compare parameters
  120.             if (MemberUtils.isMatchingConstructor(ctor, parameterTypes)) {
  121.                 // get accessible version of constructor
  122.                 ctor = getAccessibleConstructor(ctor);
  123.                 if (ctor != null) {
  124.                     MemberUtils.setAccessibleWorkaround(ctor);
  125.                     if (result == null || MemberUtils.compareConstructorFit(ctor, result, parameterTypes) < 0) {
  126.                         // temporary variable for annotation, see comment above (1)
  127.                         @SuppressWarnings("unchecked")
  128.                         final Constructor<T> constructor = (Constructor<T>) ctor;
  129.                         result = constructor;
  130.                     }
  131.                 }
  132.             }
  133.         }
  134.         return result;
  135.     }

  136.     /**
  137.      * Returns a new instance of the specified class inferring the right constructor
  138.      * from the types of the arguments.
  139.      *
  140.      * <p>This locates and calls a constructor.
  141.      * The constructor signature must match the argument types by assignment compatibility.</p>
  142.      *
  143.      * @param <T> the type to be constructed
  144.      * @param cls  the class to be constructed, not {@code null}
  145.      * @param args  the array of arguments, {@code null} treated as empty
  146.      * @return new instance of {@code cls}, not {@code null}
  147.      *
  148.      * @throws NullPointerException if {@code cls} is {@code null}
  149.      * @throws NoSuchMethodException if a matching constructor cannot be found
  150.      * @throws IllegalAccessException if invocation is not permitted by security
  151.      * @throws InvocationTargetException if an error occurs on invocation
  152.      * @throws InstantiationException if an error occurs on instantiation
  153.      * @see #invokeConstructor(Class, Object[], Class[])
  154.      */
  155.     public static <T> T invokeConstructor(final Class<T> cls, Object... args)
  156.             throws NoSuchMethodException, IllegalAccessException, InvocationTargetException,
  157.             InstantiationException {
  158.         args = ArrayUtils.nullToEmpty(args);
  159.         return invokeConstructor(cls, args, ClassUtils.toClass(args));
  160.     }

  161.     /**
  162.      * Returns a new instance of the specified class choosing the right constructor
  163.      * from the list of parameter types.
  164.      *
  165.      * <p>This locates and calls a constructor.
  166.      * The constructor signature must match the parameter types by assignment compatibility.</p>
  167.      *
  168.      * @param <T> the type to be constructed
  169.      * @param cls  the class to be constructed, not {@code null}
  170.      * @param args  the array of arguments, {@code null} treated as empty
  171.      * @param parameterTypes  the array of parameter types, {@code null} treated as empty
  172.      * @return new instance of {@code cls}, not {@code null}
  173.      *
  174.      * @throws NullPointerException if {@code cls} is {@code null}
  175.      * @throws NoSuchMethodException if a matching constructor cannot be found
  176.      * @throws IllegalAccessException if invocation is not permitted by security
  177.      * @throws InvocationTargetException if an error occurs on invocation
  178.      * @throws InstantiationException if an error occurs on instantiation
  179.      * @see Constructor#newInstance
  180.      */
  181.     public static <T> T invokeConstructor(final Class<T> cls, Object[] args, Class<?>[] parameterTypes)
  182.             throws NoSuchMethodException, IllegalAccessException, InvocationTargetException,
  183.             InstantiationException {
  184.         args = ArrayUtils.nullToEmpty(args);
  185.         parameterTypes = ArrayUtils.nullToEmpty(parameterTypes);
  186.         final Constructor<T> ctor = getMatchingAccessibleConstructor(cls, parameterTypes);
  187.         if (ctor == null) {
  188.             throw new NoSuchMethodException(
  189.                 "No such accessible constructor on object: " + cls.getName());
  190.         }
  191.         if (ctor.isVarArgs()) {
  192.             final Class<?>[] methodParameterTypes = ctor.getParameterTypes();
  193.             args = MethodUtils.getVarArgs(args, methodParameterTypes);
  194.         }
  195.         return ctor.newInstance(args);
  196.     }

  197.     /**
  198.      * Returns a new instance of the specified class inferring the right constructor
  199.      * from the types of the arguments.
  200.      *
  201.      * <p>This locates and calls a constructor.
  202.      * The constructor signature must match the argument types exactly.</p>
  203.      *
  204.      * @param <T> the type to be constructed
  205.      * @param cls the class to be constructed, not {@code null}
  206.      * @param args the array of arguments, {@code null} treated as empty
  207.      * @return new instance of {@code cls}, not {@code null}
  208.      *
  209.      * @throws NullPointerException if {@code cls} is {@code null}
  210.      * @throws NoSuchMethodException if a matching constructor cannot be found
  211.      * @throws IllegalAccessException if invocation is not permitted by security
  212.      * @throws InvocationTargetException if an error occurs on invocation
  213.      * @throws InstantiationException if an error occurs on instantiation
  214.      * @see #invokeExactConstructor(Class, Object[], Class[])
  215.      */
  216.     public static <T> T invokeExactConstructor(final Class<T> cls, Object... args)
  217.             throws NoSuchMethodException, IllegalAccessException, InvocationTargetException,
  218.             InstantiationException {
  219.         args = ArrayUtils.nullToEmpty(args);
  220.         return invokeExactConstructor(cls, args, ClassUtils.toClass(args));
  221.     }

  222.     /**
  223.      * Returns a new instance of the specified class choosing the right constructor
  224.      * from the list of parameter types.
  225.      *
  226.      * <p>This locates and calls a constructor.
  227.      * The constructor signature must match the parameter types exactly.</p>
  228.      *
  229.      * @param <T> the type to be constructed
  230.      * @param cls the class to be constructed, not {@code null}
  231.      * @param args the array of arguments, {@code null} treated as empty
  232.      * @param parameterTypes  the array of parameter types, {@code null} treated as empty
  233.      * @return new instance of {@code cls}, not {@code null}
  234.      *
  235.      * @throws NullPointerException if {@code cls} is {@code null}
  236.      * @throws NoSuchMethodException if a matching constructor cannot be found
  237.      * @throws IllegalAccessException if invocation is not permitted by security
  238.      * @throws InvocationTargetException if an error occurs on invocation
  239.      * @throws InstantiationException if an error occurs on instantiation
  240.      * @see Constructor#newInstance
  241.      */
  242.     public static <T> T invokeExactConstructor(final Class<T> cls, Object[] args,
  243.             Class<?>[] parameterTypes) throws NoSuchMethodException, IllegalAccessException,
  244.             InvocationTargetException, InstantiationException {
  245.         args = ArrayUtils.nullToEmpty(args);
  246.         parameterTypes = ArrayUtils.nullToEmpty(parameterTypes);
  247.         final Constructor<T> ctor = getAccessibleConstructor(cls, parameterTypes);
  248.         if (ctor == null) {
  249.             throw new NoSuchMethodException("No such accessible constructor on object: " + cls.getName());
  250.         }
  251.         return ctor.newInstance(args);
  252.     }

  253.     /**
  254.      * Tests whether the specified class is generally accessible, i.e. is
  255.      * declared in an entirely {@code public} manner.
  256.      * @param type to check
  257.      * @return {@code true} if {@code type} and any enclosing classes are
  258.      *         {@code public}.
  259.      */
  260.     private static boolean isAccessible(final Class<?> type) {
  261.         Class<?> cls = type;
  262.         while (cls != null) {
  263.             if (!ClassUtils.isPublic(cls)) {
  264.                 return false;
  265.             }
  266.             cls = cls.getEnclosingClass();
  267.         }
  268.         return true;
  269.     }

  270.     /**
  271.      * ConstructorUtils instances should NOT be constructed in standard
  272.      * programming. Instead, the class should be used as
  273.      * {@code ConstructorUtils.invokeConstructor(cls, args)}.
  274.      *
  275.      * <p>This constructor is {@code public} to permit tools that require a JavaBean
  276.      * instance to operate.</p>
  277.      *
  278.      * @deprecated TODO Make private in 4.0.
  279.      */
  280.     @Deprecated
  281.     public ConstructorUtils() {
  282.         // empty
  283.     }

  284. }