TypeUtils.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.AnnotatedType;
  19. import java.lang.reflect.Array;
  20. import java.lang.reflect.GenericArrayType;
  21. import java.lang.reflect.GenericDeclaration;
  22. import java.lang.reflect.ParameterizedType;
  23. import java.lang.reflect.Type;
  24. import java.lang.reflect.TypeVariable;
  25. import java.lang.reflect.WildcardType;
  26. import java.util.Arrays;
  27. import java.util.Collection;
  28. import java.util.Collections;
  29. import java.util.HashMap;
  30. import java.util.HashSet;
  31. import java.util.List;
  32. import java.util.Map;
  33. import java.util.Objects;
  34. import java.util.Set;
  35. import java.util.TreeSet;

  36. import org.apache.commons.lang3.AppendableJoiner;
  37. import org.apache.commons.lang3.ArrayUtils;
  38. import org.apache.commons.lang3.ClassUtils;
  39. import org.apache.commons.lang3.ObjectUtils;
  40. import org.apache.commons.lang3.Validate;
  41. import org.apache.commons.lang3.builder.Builder;

  42. /**
  43.  * Utility methods focusing on type inspection, particularly with regard to generics.
  44.  *
  45.  * @since 3.0
  46.  */
  47. public class TypeUtils {

  48.     /**
  49.      * GenericArrayType implementation class.
  50.      */
  51.     private static final class GenericArrayTypeImpl implements GenericArrayType {
  52.         private final Type componentType;

  53.         /**
  54.          * Constructor
  55.          *
  56.          * @param componentType of this array type
  57.          */
  58.         private GenericArrayTypeImpl(final Type componentType) {
  59.             this.componentType = componentType;
  60.         }

  61.         /**
  62.          * {@inheritDoc}
  63.          */
  64.         @Override
  65.         public boolean equals(final Object obj) {
  66.             return obj == this || obj instanceof GenericArrayType && TypeUtils.equals(this, (GenericArrayType) obj);
  67.         }

  68.         /**
  69.          * {@inheritDoc}
  70.          */
  71.         @Override
  72.         public Type getGenericComponentType() {
  73.             return componentType;
  74.         }

  75.         /**
  76.          * {@inheritDoc}
  77.          */
  78.         @Override
  79.         public int hashCode() {
  80.             int result = 67 << 4;
  81.             result |= componentType.hashCode();
  82.             return result;
  83.         }

  84.         /**
  85.          * {@inheritDoc}
  86.          */
  87.         @Override
  88.         public String toString() {
  89.             return TypeUtils.toString(this);
  90.         }
  91.     }

  92.     /**
  93.      * ParameterizedType implementation class.
  94.      */
  95.     private static final class ParameterizedTypeImpl implements ParameterizedType {
  96.         private final Class<?> raw;
  97.         private final Type useOwner;
  98.         private final Type[] typeArguments;

  99.         /**
  100.          * Constructor
  101.          *
  102.          * @param rawClass      type
  103.          * @param useOwner      owner type to use, if any
  104.          * @param typeArguments formal type arguments
  105.          */
  106.         private ParameterizedTypeImpl(final Class<?> rawClass, final Type useOwner, final Type[] typeArguments) {
  107.             this.raw = rawClass;
  108.             this.useOwner = useOwner;
  109.             this.typeArguments = Arrays.copyOf(typeArguments, typeArguments.length, Type[].class);
  110.         }

  111.         /**
  112.          * {@inheritDoc}
  113.          */
  114.         @Override
  115.         public boolean equals(final Object obj) {
  116.             return obj == this || obj instanceof ParameterizedType && TypeUtils.equals(this, (ParameterizedType) obj);
  117.         }

  118.         /**
  119.          * {@inheritDoc}
  120.          */
  121.         @Override
  122.         public Type[] getActualTypeArguments() {
  123.             return typeArguments.clone();
  124.         }

  125.         /**
  126.          * {@inheritDoc}
  127.          */
  128.         @Override
  129.         public Type getOwnerType() {
  130.             return useOwner;
  131.         }

  132.         /**
  133.          * {@inheritDoc}
  134.          */
  135.         @Override
  136.         public Type getRawType() {
  137.             return raw;
  138.         }

  139.         /**
  140.          * {@inheritDoc}
  141.          */
  142.         @Override
  143.         public int hashCode() {
  144.             int result = 71 << 4;
  145.             result |= raw.hashCode();
  146.             result <<= 4;
  147.             result |= Objects.hashCode(useOwner);
  148.             result <<= 8;
  149.             result |= Arrays.hashCode(typeArguments);
  150.             return result;
  151.         }

  152.         /**
  153.          * {@inheritDoc}
  154.          */
  155.         @Override
  156.         public String toString() {
  157.             return TypeUtils.toString(this);
  158.         }
  159.     }

  160.     /**
  161.      * {@link WildcardType} builder.
  162.      *
  163.      * @since 3.2
  164.      */
  165.     public static class WildcardTypeBuilder implements Builder<WildcardType> {
  166.         private Type[] upperBounds;

  167.         private Type[] lowerBounds;

  168.         /**
  169.          * Constructor
  170.          */
  171.         private WildcardTypeBuilder() {
  172.         }

  173.         /**
  174.          * {@inheritDoc}
  175.          */
  176.         @Override
  177.         public WildcardType build() {
  178.             return new WildcardTypeImpl(upperBounds, lowerBounds);
  179.         }

  180.         /**
  181.          * Specify lower bounds of the wildcard type to build.
  182.          *
  183.          * @param bounds to set
  184.          * @return {@code this}
  185.          */
  186.         public WildcardTypeBuilder withLowerBounds(final Type... bounds) {
  187.             this.lowerBounds = bounds;
  188.             return this;
  189.         }

  190.         /**
  191.          * Specify upper bounds of the wildcard type to build.
  192.          *
  193.          * @param bounds to set
  194.          * @return {@code this}
  195.          */
  196.         public WildcardTypeBuilder withUpperBounds(final Type... bounds) {
  197.             this.upperBounds = bounds;
  198.             return this;
  199.         }
  200.     }

  201.     /**
  202.      * WildcardType implementation class.
  203.      */
  204.     private static final class WildcardTypeImpl implements WildcardType {
  205.         private final Type[] upperBounds;
  206.         private final Type[] lowerBounds;

  207.         /**
  208.          * Constructor
  209.          *
  210.          * @param upperBounds of this type
  211.          * @param lowerBounds of this type
  212.          */
  213.         private WildcardTypeImpl(final Type[] upperBounds, final Type[] lowerBounds) {
  214.             this.upperBounds = ObjectUtils.defaultIfNull(upperBounds, ArrayUtils.EMPTY_TYPE_ARRAY);
  215.             this.lowerBounds = ObjectUtils.defaultIfNull(lowerBounds, ArrayUtils.EMPTY_TYPE_ARRAY);
  216.         }

  217.         /**
  218.          * {@inheritDoc}
  219.          */
  220.         @Override
  221.         public boolean equals(final Object obj) {
  222.             return obj == this || obj instanceof WildcardType && TypeUtils.equals(this, (WildcardType) obj);
  223.         }

  224.         /**
  225.          * {@inheritDoc}
  226.          */
  227.         @Override
  228.         public Type[] getLowerBounds() {
  229.             return lowerBounds.clone();
  230.         }

  231.         /**
  232.          * {@inheritDoc}
  233.          */
  234.         @Override
  235.         public Type[] getUpperBounds() {
  236.             return upperBounds.clone();
  237.         }

  238.         /**
  239.          * {@inheritDoc}
  240.          */
  241.         @Override
  242.         public int hashCode() {
  243.             int result = 73 << 8;
  244.             result |= Arrays.hashCode(upperBounds);
  245.             result <<= 8;
  246.             result |= Arrays.hashCode(lowerBounds);
  247.             return result;
  248.         }

  249.         /**
  250.          * {@inheritDoc}
  251.          */
  252.         @Override
  253.         public String toString() {
  254.             return TypeUtils.toString(this);
  255.         }
  256.     }

  257.     /**
  258.      * Ampersand sign joiner.
  259.      */
  260.     // @formatter:off
  261.     private static final AppendableJoiner<Type> AMP_JOINER = AppendableJoiner.<Type>builder()
  262.             .setDelimiter(" & ")
  263.             .setElementAppender((a, e) -> a.append(TypeUtils.toString(e)))
  264.             .get();
  265.     // @formatter:on

  266.     /**
  267.      * Method classToString joiner.
  268.      */
  269.     // @formatter:off
  270.     private static final AppendableJoiner<TypeVariable<Class<?>>> CTJ_JOINER = AppendableJoiner.<TypeVariable<Class<?>>>builder()
  271.         .setDelimiter(", ")
  272.         .setElementAppender((a, e) -> a.append(TypeUtils.anyToString(e)))
  273.         .get();
  274.     // @formatter:on

  275.     /**
  276.      * Greater than and lesser than sign joiner.
  277.      */
  278.     // @formatter:off
  279.     private static final AppendableJoiner<Object> GT_JOINER = AppendableJoiner.builder()
  280.             .setPrefix("<")
  281.             .setSuffix(">")
  282.             .setDelimiter(", ")
  283.             .setElementAppender((a, e) -> a.append(TypeUtils.anyToString(e)))
  284.             .get();
  285.     // @formatter:on

  286.     /**
  287.      * A wildcard instance matching {@code ?}.
  288.      *
  289.      * @since 3.2
  290.      */
  291.     public static final WildcardType WILDCARD_ALL = wildcardType().withUpperBounds(Object.class).build();

  292.     private static <T> String anyToString(final T object) {
  293.         return object instanceof Type ? toString((Type) object) : object.toString();
  294.     }

  295.     private static void appendRecursiveTypes(final StringBuilder builder, final int[] recursiveTypeIndexes, final Type[] argumentTypes) {
  296.         for (int i = 0; i < recursiveTypeIndexes.length; i++) {
  297.             // toString() or SO
  298.             GT_JOINER.join(builder, argumentTypes[i].toString());
  299.         }
  300.         final Type[] argumentsFiltered = ArrayUtils.removeAll(argumentTypes, recursiveTypeIndexes);
  301.         if (argumentsFiltered.length > 0) {
  302.             GT_JOINER.join(builder, (Object[]) argumentsFiltered);
  303.         }
  304.     }

  305.     /**
  306.      * Formats a {@link Class} as a {@link String}.
  307.      *
  308.      * @param cls {@link Class} to format
  309.      * @return String
  310.      */
  311.     private static <T> String classToString(final Class<T> cls) {
  312.         if (cls.isArray()) {
  313.             return toString(cls.getComponentType()) + "[]";
  314.         }
  315.         if (isCyclical(cls)) {
  316.             return cls.getSimpleName() + "(cycle)";
  317.         }
  318.         final StringBuilder buf = new StringBuilder();
  319.         if (cls.getEnclosingClass() != null) {
  320.             buf.append(classToString(cls.getEnclosingClass())).append('.').append(cls.getSimpleName());
  321.         } else {
  322.             buf.append(cls.getName());
  323.         }
  324.         if (cls.getTypeParameters().length > 0) {
  325.             // AppendableJoiner.joinSB(buf, null, null, ", ", TypeUtils::anyToString, cls.getTypeParameters());
  326.             CTJ_JOINER.join(buf, (TypeVariable[]) cls.getTypeParameters());
  327.         }
  328.         return buf.toString();
  329.     }

  330.     /**
  331.      * Tests, recursively, whether any of the type parameters associated with {@code type} are bound to variables.
  332.      *
  333.      * @param type the type to check for type variables
  334.      * @return boolean
  335.      * @since 3.2
  336.      */
  337.     public static boolean containsTypeVariables(final Type type) {
  338.         if (type instanceof TypeVariable<?>) {
  339.             return true;
  340.         }
  341.         if (type instanceof Class<?>) {
  342.             return ((Class<?>) type).getTypeParameters().length > 0;
  343.         }
  344.         if (type instanceof ParameterizedType) {
  345.             for (final Type arg : ((ParameterizedType) type).getActualTypeArguments()) {
  346.                 if (containsTypeVariables(arg)) {
  347.                     return true;
  348.                 }
  349.             }
  350.             return false;
  351.         }
  352.         if (type instanceof WildcardType) {
  353.             final WildcardType wild = (WildcardType) type;
  354.             return containsTypeVariables(getImplicitLowerBounds(wild)[0]) || containsTypeVariables(getImplicitUpperBounds(wild)[0]);
  355.         }
  356.         if (type instanceof GenericArrayType) {
  357.             return containsTypeVariables(((GenericArrayType) type).getGenericComponentType());
  358.         }
  359.         return false;
  360.     }

  361.     private static boolean containsVariableTypeSameParametrizedTypeBound(final TypeVariable<?> typeVariable, final ParameterizedType parameterizedType) {
  362.         return ArrayUtils.contains(typeVariable.getBounds(), parameterizedType);
  363.     }

  364.     /**
  365.      * Tries to determine the type arguments of a class/interface based on a super parameterized type's type arguments. This method is the inverse of
  366.      * {@link #getTypeArguments(Type, Class)} which gets a class/interface's type arguments based on a subtype. It is far more limited in determining the type
  367.      * arguments for the subject class's type variables in that it can only determine those parameters that map from the subject {@link Class} object to the
  368.      * supertype.
  369.      *
  370.      * <p>
  371.      * Example: {@link java.util.TreeSet TreeSet} sets its parameter as the parameter for {@link java.util.NavigableSet NavigableSet}, which in turn sets the
  372.      * parameter of {@link java.util.SortedSet}, which in turn sets the parameter of {@link Set}, which in turn sets the parameter of
  373.      * {@link java.util.Collection}, which in turn sets the parameter of {@link Iterable}. Since {@link TreeSet}'s parameter maps (indirectly) to
  374.      * {@link Iterable}'s parameter, it will be able to determine that based on the super type {@code Iterable<? extends
  375.      * Map<Integer, ? extends Collection<?>>>}, the parameter of {@link TreeSet} is {@code ? extends Map<Integer, ? extends
  376.      * Collection<?>>}.
  377.      * </p>
  378.      *
  379.      * @param cls                    the class whose type parameters are to be determined, not {@code null}
  380.      * @param superParameterizedType the super type from which {@code cls}'s type arguments are to be determined, not {@code null}
  381.      * @return a {@link Map} of the type assignments that could be determined for the type variables in each type in the inheritance hierarchy from {@code type}
  382.      *         to {@code toClass} inclusive.
  383.      * @throws NullPointerException if either {@code cls} or {@code superParameterizedType} is {@code null}
  384.      */
  385.     public static Map<TypeVariable<?>, Type> determineTypeArguments(final Class<?> cls, final ParameterizedType superParameterizedType) {
  386.         Objects.requireNonNull(cls, "cls");
  387.         Objects.requireNonNull(superParameterizedType, "superParameterizedType");

  388.         final Class<?> superClass = getRawType(superParameterizedType);

  389.         // compatibility check
  390.         if (!isAssignable(cls, superClass)) {
  391.             return null;
  392.         }

  393.         if (cls.equals(superClass)) {
  394.             return getTypeArguments(superParameterizedType, superClass, null);
  395.         }

  396.         // get the next class in the inheritance hierarchy
  397.         final Type midType = getClosestParentType(cls, superClass);

  398.         // can only be a class or a parameterized type
  399.         if (midType instanceof Class<?>) {
  400.             return determineTypeArguments((Class<?>) midType, superParameterizedType);
  401.         }

  402.         final ParameterizedType midParameterizedType = (ParameterizedType) midType;
  403.         final Class<?> midClass = getRawType(midParameterizedType);
  404.         // get the type variables of the mid class that map to the type
  405.         // arguments of the super class
  406.         final Map<TypeVariable<?>, Type> typeVarAssigns = determineTypeArguments(midClass, superParameterizedType);
  407.         // map the arguments of the mid type to the class type variables
  408.         mapTypeVariablesToArguments(cls, midParameterizedType, typeVarAssigns);

  409.         return typeVarAssigns;
  410.     }

  411.     /**
  412.      * Tests whether {@code t} equals {@code a}.
  413.      *
  414.      * @param genericArrayType LHS
  415.      * @param type             RHS
  416.      * @return boolean
  417.      */
  418.     private static boolean equals(final GenericArrayType genericArrayType, final Type type) {
  419.         return type instanceof GenericArrayType && equals(genericArrayType.getGenericComponentType(), ((GenericArrayType) type).getGenericComponentType());
  420.     }

  421.     /**
  422.      * Tests whether {@code t} equals {@code p}.
  423.      *
  424.      * @param parameterizedType LHS
  425.      * @param type              RHS
  426.      * @return boolean
  427.      */
  428.     private static boolean equals(final ParameterizedType parameterizedType, final Type type) {
  429.         if (type instanceof ParameterizedType) {
  430.             final ParameterizedType other = (ParameterizedType) type;
  431.             if (equals(parameterizedType.getRawType(), other.getRawType()) && equals(parameterizedType.getOwnerType(), other.getOwnerType())) {
  432.                 return equals(parameterizedType.getActualTypeArguments(), other.getActualTypeArguments());
  433.             }
  434.         }
  435.         return false;
  436.     }

  437.     /**
  438.      * Tests equality of types.
  439.      *
  440.      * @param type1 the first type
  441.      * @param type2 the second type
  442.      * @return boolean
  443.      * @since 3.2
  444.      */
  445.     public static boolean equals(final Type type1, final Type type2) {
  446.         if (Objects.equals(type1, type2)) {
  447.             return true;
  448.         }
  449.         if (type1 instanceof ParameterizedType) {
  450.             return equals((ParameterizedType) type1, type2);
  451.         }
  452.         if (type1 instanceof GenericArrayType) {
  453.             return equals((GenericArrayType) type1, type2);
  454.         }
  455.         if (type1 instanceof WildcardType) {
  456.             return equals((WildcardType) type1, type2);
  457.         }
  458.         return false;
  459.     }

  460.     /**
  461.      * Tests whether {@code t1} equals {@code t2}.
  462.      *
  463.      * @param type1 LHS
  464.      * @param type2 RHS
  465.      * @return boolean
  466.      */
  467.     private static boolean equals(final Type[] type1, final Type[] type2) {
  468.         if (type1.length == type2.length) {
  469.             for (int i = 0; i < type1.length; i++) {
  470.                 if (!equals(type1[i], type2[i])) {
  471.                     return false;
  472.                 }
  473.             }
  474.             return true;
  475.         }
  476.         return false;
  477.     }

  478.     /**
  479.      * Tests whether {@code t} equals {@code w}.
  480.      *
  481.      * @param wildcardType LHS
  482.      * @param type         RHS
  483.      * @return boolean
  484.      */
  485.     private static boolean equals(final WildcardType wildcardType, final Type type) {
  486.         if (type instanceof WildcardType) {
  487.             final WildcardType other = (WildcardType) type;
  488.             return equals(getImplicitLowerBounds(wildcardType), getImplicitLowerBounds(other))
  489.                     && equals(getImplicitUpperBounds(wildcardType), getImplicitUpperBounds(other));
  490.         }
  491.         return false;
  492.     }

  493.     /**
  494.      * Helper method to establish the formal parameters for a parameterized type.
  495.      *
  496.      * @param mappings  map containing the assignments
  497.      * @param variables expected map keys
  498.      * @return array of map values corresponding to specified keys
  499.      */
  500.     private static Type[] extractTypeArgumentsFrom(final Map<TypeVariable<?>, Type> mappings, final TypeVariable<?>[] variables) {
  501.         final Type[] result = new Type[variables.length];
  502.         int index = 0;
  503.         for (final TypeVariable<?> var : variables) {
  504.             Validate.isTrue(mappings.containsKey(var), "missing argument mapping for %s", toString(var));
  505.             result[index++] = mappings.get(var);
  506.         }
  507.         return result;
  508.     }

  509.     private static int[] findRecursiveTypes(final ParameterizedType parameterizedType) {
  510.         final Type[] filteredArgumentTypes = Arrays.copyOf(parameterizedType.getActualTypeArguments(), parameterizedType.getActualTypeArguments().length);
  511.         int[] indexesToRemove = {};
  512.         for (int i = 0; i < filteredArgumentTypes.length; i++) {
  513.             if (filteredArgumentTypes[i] instanceof TypeVariable<?>
  514.                     && containsVariableTypeSameParametrizedTypeBound((TypeVariable<?>) filteredArgumentTypes[i], parameterizedType)) {
  515.                 indexesToRemove = ArrayUtils.add(indexesToRemove, i);
  516.             }
  517.         }
  518.         return indexesToRemove;
  519.     }

  520.     /**
  521.      * Creates a generic array type instance.
  522.      *
  523.      * @param componentType the type of the elements of the array. For example the component type of {@code boolean[]} is {@code boolean}
  524.      * @return {@link GenericArrayType}
  525.      * @since 3.2
  526.      */
  527.     public static GenericArrayType genericArrayType(final Type componentType) {
  528.         return new GenericArrayTypeImpl(Objects.requireNonNull(componentType, "componentType"));
  529.     }

  530.     /**
  531.      * Formats a {@link GenericArrayType} as a {@link String}.
  532.      *
  533.      * @param genericArrayType {@link GenericArrayType} to format
  534.      * @return String
  535.      */
  536.     private static String genericArrayTypeToString(final GenericArrayType genericArrayType) {
  537.         return String.format("%s[]", toString(genericArrayType.getGenericComponentType()));
  538.     }

  539.     /**
  540.      * Gets the array component type of {@code type}.
  541.      *
  542.      * @param type the type to be checked
  543.      * @return component type or null if type is not an array type
  544.      */
  545.     public static Type getArrayComponentType(final Type type) {
  546.         if (type instanceof Class<?>) {
  547.             final Class<?> cls = (Class<?>) type;
  548.             return cls.isArray() ? cls.getComponentType() : null;
  549.         }
  550.         if (type instanceof GenericArrayType) {
  551.             return ((GenericArrayType) type).getGenericComponentType();
  552.         }
  553.         return null;
  554.     }

  555.     /**
  556.      * Gets the closest parent type to the super class specified by {@code superClass}.
  557.      *
  558.      * @param cls        the class in question
  559.      * @param superClass the super class
  560.      * @return the closes parent type
  561.      */
  562.     private static Type getClosestParentType(final Class<?> cls, final Class<?> superClass) {
  563.         // only look at the interfaces if the super class is also an interface
  564.         if (superClass.isInterface()) {
  565.             // get the generic interfaces of the subject class
  566.             final Type[] interfaceTypes = cls.getGenericInterfaces();
  567.             // will hold the best generic interface match found
  568.             Type genericInterface = null;

  569.             // find the interface closest to the super class
  570.             for (final Type midType : interfaceTypes) {
  571.                 final Class<?> midClass;

  572.                 if (midType instanceof ParameterizedType) {
  573.                     midClass = getRawType((ParameterizedType) midType);
  574.                 } else if (midType instanceof Class<?>) {
  575.                     midClass = (Class<?>) midType;
  576.                 } else {
  577.                     throw new IllegalStateException("Unexpected generic" + " interface type found: " + midType);
  578.                 }

  579.                 // check if this interface is further up the inheritance chain
  580.                 // than the previously found match
  581.                 if (isAssignable(midClass, superClass) && isAssignable(genericInterface, (Type) midClass)) {
  582.                     genericInterface = midType;
  583.                 }
  584.             }

  585.             // found a match?
  586.             if (genericInterface != null) {
  587.                 return genericInterface;
  588.             }
  589.         }

  590.         // none of the interfaces were descendants of the target class, so the
  591.         // super class has to be one, instead
  592.         return cls.getGenericSuperclass();
  593.     }

  594.     /**
  595.      * Gets an array containing the sole type of {@link Object} if {@link TypeVariable#getBounds()} returns an empty array. Otherwise, it returns the result of
  596.      * {@link TypeVariable#getBounds()} passed into {@link #normalizeUpperBounds}.
  597.      *
  598.      * @param typeVariable the subject type variable, not {@code null}
  599.      * @return a non-empty array containing the bounds of the type variable.
  600.      * @throws NullPointerException if {@code typeVariable} is {@code null}
  601.      */
  602.     public static Type[] getImplicitBounds(final TypeVariable<?> typeVariable) {
  603.         Objects.requireNonNull(typeVariable, "typeVariable");
  604.         final Type[] bounds = typeVariable.getBounds();

  605.         return bounds.length == 0 ? new Type[] { Object.class } : normalizeUpperBounds(bounds);
  606.     }

  607.     /**
  608.      * Gets an array containing a single value of {@code null} if {@link WildcardType#getLowerBounds()} returns an empty array. Otherwise, it returns the result
  609.      * of {@link WildcardType#getLowerBounds()}.
  610.      *
  611.      * @param wildcardType the subject wildcard type, not {@code null}
  612.      * @return a non-empty array containing the lower bounds of the wildcard type.
  613.      * @throws NullPointerException if {@code wildcardType} is {@code null}
  614.      */
  615.     public static Type[] getImplicitLowerBounds(final WildcardType wildcardType) {
  616.         Objects.requireNonNull(wildcardType, "wildcardType");
  617.         final Type[] bounds = wildcardType.getLowerBounds();

  618.         return bounds.length == 0 ? new Type[] { null } : bounds;
  619.     }

  620.     /**
  621.      * Gets an array containing the sole value of {@link Object} if {@link WildcardType#getUpperBounds()} returns an empty array. Otherwise, it returns the
  622.      * result of {@link WildcardType#getUpperBounds()} passed into {@link #normalizeUpperBounds}.
  623.      *
  624.      * @param wildcardType the subject wildcard type, not {@code null}
  625.      * @return a non-empty array containing the upper bounds of the wildcard type.
  626.      * @throws NullPointerException if {@code wildcardType} is {@code null}
  627.      */
  628.     public static Type[] getImplicitUpperBounds(final WildcardType wildcardType) {
  629.         Objects.requireNonNull(wildcardType, "wildcardType");
  630.         final Type[] bounds = wildcardType.getUpperBounds();

  631.         return bounds.length == 0 ? new Type[] { Object.class } : normalizeUpperBounds(bounds);
  632.     }

  633.     /**
  634.      * Transforms the passed in type to a {@link Class} object. Type-checking method of convenience.
  635.      *
  636.      * @param parameterizedType the type to be converted
  637.      * @return the corresponding {@link Class} object
  638.      * @throws IllegalStateException if the conversion fails
  639.      */
  640.     private static Class<?> getRawType(final ParameterizedType parameterizedType) {
  641.         final Type rawType = parameterizedType.getRawType();

  642.         // check if raw type is a Class object
  643.         // not currently necessary, but since the return type is Type instead of
  644.         // Class, there's enough reason to believe that future versions of Java
  645.         // may return other Type implementations. And type-safety checking is
  646.         // rarely a bad idea.
  647.         if (!(rawType instanceof Class<?>)) {
  648.             throw new IllegalStateException("Wait... What!? Type of rawType: " + rawType);
  649.         }

  650.         return (Class<?>) rawType;
  651.     }

  652.     /**
  653.      * Gets the raw type of a Java type, given its context. Primarily for use with {@link TypeVariable}s and {@link GenericArrayType}s, or when you do not know
  654.      * the runtime type of {@code type}: if you know you have a {@link Class} instance, it is already raw; if you know you have a {@link ParameterizedType}, its
  655.      * raw type is only a method call away.
  656.      *
  657.      * @param type          to resolve
  658.      * @param assigningType type to be resolved against
  659.      * @return the resolved {@link Class} object or {@code null} if the type could not be resolved
  660.      */
  661.     public static Class<?> getRawType(final Type type, final Type assigningType) {
  662.         if (type instanceof Class<?>) {
  663.             // it is raw, no problem
  664.             return (Class<?>) type;
  665.         }

  666.         if (type instanceof ParameterizedType) {
  667.             // simple enough to get the raw type of a ParameterizedType
  668.             return getRawType((ParameterizedType) type);
  669.         }

  670.         if (type instanceof TypeVariable<?>) {
  671.             if (assigningType == null) {
  672.                 return null;
  673.             }

  674.             // get the entity declaring this type variable
  675.             final Object genericDeclaration = ((TypeVariable<?>) type).getGenericDeclaration();

  676.             // can't get the raw type of a method- or constructor-declared type
  677.             // variable
  678.             if (!(genericDeclaration instanceof Class<?>)) {
  679.                 return null;
  680.             }

  681.             // get the type arguments for the declaring class/interface based
  682.             // on the enclosing type
  683.             final Map<TypeVariable<?>, Type> typeVarAssigns = getTypeArguments(assigningType, (Class<?>) genericDeclaration);

  684.             // enclosingType has to be a subclass (or subinterface) of the
  685.             // declaring type
  686.             if (typeVarAssigns == null) {
  687.                 return null;
  688.             }

  689.             // get the argument assigned to this type variable
  690.             final Type typeArgument = typeVarAssigns.get(type);

  691.             if (typeArgument == null) {
  692.                 return null;
  693.             }

  694.             // get the argument for this type variable
  695.             return getRawType(typeArgument, assigningType);
  696.         }

  697.         if (type instanceof GenericArrayType) {
  698.             // get raw component type
  699.             final Class<?> rawComponentType = getRawType(((GenericArrayType) type).getGenericComponentType(), assigningType);

  700.             // create array type from raw component type and return its class
  701.             return rawComponentType != null ? Array.newInstance(rawComponentType, 0).getClass() : null;
  702.         }

  703.         // (hand-waving) this is not the method you're looking for
  704.         if (type instanceof WildcardType) {
  705.             return null;
  706.         }

  707.         throw new IllegalArgumentException("unknown type: " + type);
  708.     }

  709.     /**
  710.      * Gets a map of the type arguments of a class in the context of {@code toClass}.
  711.      *
  712.      * @param cls               the class in question
  713.      * @param toClass           the context class
  714.      * @param subtypeVarAssigns a map with type variables
  715.      * @return the {@link Map} with type arguments
  716.      */
  717.     private static Map<TypeVariable<?>, Type> getTypeArguments(Class<?> cls, final Class<?> toClass, final Map<TypeVariable<?>, Type> subtypeVarAssigns) {
  718.         // make sure they're assignable
  719.         if (!isAssignable(cls, toClass)) {
  720.             return null;
  721.         }

  722.         // can't work with primitives
  723.         if (cls.isPrimitive()) {
  724.             // both classes are primitives?
  725.             if (toClass.isPrimitive()) {
  726.                 // dealing with widening here. No type arguments to be
  727.                 // harvested with these two types.
  728.                 return new HashMap<>();
  729.             }

  730.             // work with wrapper the wrapper class instead of the primitive
  731.             cls = ClassUtils.primitiveToWrapper(cls);
  732.         }

  733.         // create a copy of the incoming map, or an empty one if it's null
  734.         final HashMap<TypeVariable<?>, Type> typeVarAssigns = subtypeVarAssigns == null ? new HashMap<>() : new HashMap<>(subtypeVarAssigns);

  735.         // has target class been reached?
  736.         if (toClass.equals(cls)) {
  737.             return typeVarAssigns;
  738.         }

  739.         // walk the inheritance hierarchy until the target class is reached
  740.         return getTypeArguments(getClosestParentType(cls, toClass), toClass, typeVarAssigns);
  741.     }

  742.     /**
  743.      * Gets all the type arguments for this parameterized type including owner hierarchy arguments such as {@code Outer<K, V>.Inner<T>.DeepInner<E>} . The
  744.      * arguments are returned in a {@link Map} specifying the argument type for each {@link TypeVariable}.
  745.      *
  746.      * @param type specifies the subject parameterized type from which to harvest the parameters.
  747.      * @return a {@link Map} of the type arguments to their respective type variables.
  748.      */
  749.     public static Map<TypeVariable<?>, Type> getTypeArguments(final ParameterizedType type) {
  750.         return getTypeArguments(type, getRawType(type), null);
  751.     }

  752.     /**
  753.      * Gets a map of the type arguments of a parameterized type in the context of {@code toClass}.
  754.      *
  755.      * @param parameterizedType the parameterized type
  756.      * @param toClass           the class
  757.      * @param subtypeVarAssigns a map with type variables
  758.      * @return the {@link Map} with type arguments
  759.      */
  760.     private static Map<TypeVariable<?>, Type> getTypeArguments(final ParameterizedType parameterizedType, final Class<?> toClass,
  761.             final Map<TypeVariable<?>, Type> subtypeVarAssigns) {
  762.         final Class<?> cls = getRawType(parameterizedType);

  763.         // make sure they're assignable
  764.         if (!isAssignable(cls, toClass)) {
  765.             return null;
  766.         }

  767.         final Type ownerType = parameterizedType.getOwnerType();
  768.         final Map<TypeVariable<?>, Type> typeVarAssigns;

  769.         if (ownerType instanceof ParameterizedType) {
  770.             // get the owner type arguments first
  771.             final ParameterizedType parameterizedOwnerType = (ParameterizedType) ownerType;
  772.             typeVarAssigns = getTypeArguments(parameterizedOwnerType, getRawType(parameterizedOwnerType), subtypeVarAssigns);
  773.         } else {
  774.             // no owner, prep the type variable assignments map
  775.             typeVarAssigns = subtypeVarAssigns == null ? new HashMap<>() : new HashMap<>(subtypeVarAssigns);
  776.         }

  777.         // get the subject parameterized type's arguments
  778.         final Type[] typeArgs = parameterizedType.getActualTypeArguments();
  779.         // and get the corresponding type variables from the raw class
  780.         final TypeVariable<?>[] typeParams = cls.getTypeParameters();

  781.         // map the arguments to their respective type variables
  782.         for (int i = 0; i < typeParams.length; i++) {
  783.             final Type typeArg = typeArgs[i];
  784.             typeVarAssigns.put(typeParams[i], typeVarAssigns.getOrDefault(typeArg, typeArg));
  785.         }

  786.         if (toClass.equals(cls)) {
  787.             // target class has been reached. Done.
  788.             return typeVarAssigns;
  789.         }

  790.         // walk the inheritance hierarchy until the target class is reached
  791.         return getTypeArguments(getClosestParentType(cls, toClass), toClass, typeVarAssigns);
  792.     }

  793.     /**
  794.      * Gets the type arguments of a class/interface based on a subtype. For instance, this method will determine that both of the parameters for the interface
  795.      * {@link Map} are {@link Object} for the subtype {@link java.util.Properties Properties} even though the subtype does not directly implement the
  796.      * {@link Map} interface.
  797.      *
  798.      * <p>
  799.      * This method returns {@code null} if {@code type} is not assignable to {@code toClass}. It returns an empty map if none of the classes or interfaces in
  800.      * its inheritance hierarchy specify any type arguments.
  801.      * </p>
  802.      *
  803.      * <p>
  804.      * A side effect of this method is that it also retrieves the type arguments for the classes and interfaces that are part of the hierarchy between
  805.      * {@code type} and {@code toClass}. So with the above example, this method will also determine that the type arguments for {@link java.util.Hashtable
  806.      * Hashtable} are also both {@link Object}. In cases where the interface specified by {@code toClass} is (indirectly) implemented more than once (e.g. where
  807.      * {@code toClass} specifies the interface {@link Iterable Iterable} and {@code type} specifies a parameterized type that implements both
  808.      * {@link java.util.Set Set} and {@link java.util.Collection Collection}), this method will look at the inheritance hierarchy of only one of the
  809.      * implementations/subclasses; the first interface encountered that isn't a subinterface to one of the others in the {@code type} to {@code toClass}
  810.      * hierarchy.
  811.      * </p>
  812.      *
  813.      * @param type    the type from which to determine the type parameters of {@code toClass}
  814.      * @param toClass the class whose type parameters are to be determined based on the subtype {@code type}
  815.      * @return a {@link Map} of the type assignments for the type variables in each type in the inheritance hierarchy from {@code type} to {@code toClass}
  816.      *         inclusive.
  817.      */
  818.     public static Map<TypeVariable<?>, Type> getTypeArguments(final Type type, final Class<?> toClass) {
  819.         return getTypeArguments(type, toClass, null);
  820.     }

  821.     /**
  822.      * Gets a map of the type arguments of {@code type} in the context of {@code toClass}.
  823.      *
  824.      * @param type              the type in question
  825.      * @param toClass           the class
  826.      * @param subtypeVarAssigns a map with type variables
  827.      * @return the {@link Map} with type arguments
  828.      */
  829.     private static Map<TypeVariable<?>, Type> getTypeArguments(final Type type, final Class<?> toClass, final Map<TypeVariable<?>, Type> subtypeVarAssigns) {
  830.         if (type instanceof Class<?>) {
  831.             return getTypeArguments((Class<?>) type, toClass, subtypeVarAssigns);
  832.         }

  833.         if (type instanceof ParameterizedType) {
  834.             return getTypeArguments((ParameterizedType) type, toClass, subtypeVarAssigns);
  835.         }

  836.         if (type instanceof GenericArrayType) {
  837.             return getTypeArguments(((GenericArrayType) type).getGenericComponentType(), toClass.isArray() ? toClass.getComponentType() : toClass,
  838.                     subtypeVarAssigns);
  839.         }

  840.         // since wildcard types are not assignable to classes, should this just
  841.         // return null?
  842.         if (type instanceof WildcardType) {
  843.             for (final Type bound : getImplicitUpperBounds((WildcardType) type)) {
  844.                 // find the first bound that is assignable to the target class
  845.                 if (isAssignable(bound, toClass)) {
  846.                     return getTypeArguments(bound, toClass, subtypeVarAssigns);
  847.                 }
  848.             }

  849.             return null;
  850.         }

  851.         if (type instanceof TypeVariable<?>) {
  852.             for (final Type bound : getImplicitBounds((TypeVariable<?>) type)) {
  853.                 // find the first bound that is assignable to the target class
  854.                 if (isAssignable(bound, toClass)) {
  855.                     return getTypeArguments(bound, toClass, subtypeVarAssigns);
  856.                 }
  857.             }

  858.             return null;
  859.         }
  860.         throw new IllegalStateException("found an unhandled type: " + type);
  861.     }

  862.     /**
  863.      * Tests whether the specified type denotes an array type.
  864.      *
  865.      * @param type the type to be checked
  866.      * @return {@code true} if {@code type} is an array class or a {@link GenericArrayType}.
  867.      */
  868.     public static boolean isArrayType(final Type type) {
  869.         return type instanceof GenericArrayType || type instanceof Class<?> && ((Class<?>) type).isArray();
  870.     }

  871.     /**
  872.      * Tests if the subject type may be implicitly cast to the target class following the Java generics rules.
  873.      *
  874.      * @param type    the subject type to be assigned to the target type
  875.      * @param toClass the target class
  876.      * @return {@code true} if {@code type} is assignable to {@code toClass}.
  877.      */
  878.     private static boolean isAssignable(final Type type, final Class<?> toClass) {
  879.         if (type == null) {
  880.             // consistency with ClassUtils.isAssignable() behavior
  881.             return toClass == null || !toClass.isPrimitive();
  882.         }

  883.         // only a null type can be assigned to null type which
  884.         // would have cause the previous to return true
  885.         if (toClass == null) {
  886.             return false;
  887.         }

  888.         // all types are assignable to themselves
  889.         if (toClass.equals(type)) {
  890.             return true;
  891.         }

  892.         if (type instanceof Class<?>) {
  893.             // just comparing two classes
  894.             return ClassUtils.isAssignable((Class<?>) type, toClass);
  895.         }

  896.         if (type instanceof ParameterizedType) {
  897.             // only have to compare the raw type to the class
  898.             return isAssignable(getRawType((ParameterizedType) type), toClass);
  899.         }

  900.         // *
  901.         if (type instanceof TypeVariable<?>) {
  902.             // if any of the bounds are assignable to the class, then the
  903.             // type is assignable to the class.
  904.             for (final Type bound : ((TypeVariable<?>) type).getBounds()) {
  905.                 if (isAssignable(bound, toClass)) {
  906.                     return true;
  907.                 }
  908.             }

  909.             return false;
  910.         }

  911.         // the only classes to which a generic array type can be assigned
  912.         // are class Object and array classes
  913.         if (type instanceof GenericArrayType) {
  914.             return toClass.equals(Object.class)
  915.                     || toClass.isArray() && isAssignable(((GenericArrayType) type).getGenericComponentType(), toClass.getComponentType());
  916.         }

  917.         // wildcard types are not assignable to a class (though one would think
  918.         // "? super Object" would be assignable to Object)
  919.         if (type instanceof WildcardType) {
  920.             return false;
  921.         }

  922.         throw new IllegalStateException("found an unhandled type: " + type);
  923.     }

  924.     /**
  925.      * Tests if the subject type may be implicitly cast to the target generic array type following the Java generics rules.
  926.      *
  927.      * @param type               the subject type to be assigned to the target type
  928.      * @param toGenericArrayType the target generic array type
  929.      * @param typeVarAssigns     a map with type variables
  930.      * @return {@code true} if {@code type} is assignable to {@code toGenericArrayType}.
  931.      */
  932.     private static boolean isAssignable(final Type type, final GenericArrayType toGenericArrayType, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  933.         if (type == null) {
  934.             return true;
  935.         }

  936.         // only a null type can be assigned to null type which
  937.         // would have cause the previous to return true
  938.         if (toGenericArrayType == null) {
  939.             return false;
  940.         }

  941.         // all types are assignable to themselves
  942.         if (toGenericArrayType.equals(type)) {
  943.             return true;
  944.         }

  945.         final Type toComponentType = toGenericArrayType.getGenericComponentType();

  946.         if (type instanceof Class<?>) {
  947.             final Class<?> cls = (Class<?>) type;

  948.             // compare the component types
  949.             return cls.isArray() && isAssignable(cls.getComponentType(), toComponentType, typeVarAssigns);
  950.         }

  951.         if (type instanceof GenericArrayType) {
  952.             // compare the component types
  953.             return isAssignable(((GenericArrayType) type).getGenericComponentType(), toComponentType, typeVarAssigns);
  954.         }

  955.         if (type instanceof WildcardType) {
  956.             // so long as one of the upper bounds is assignable, it's good
  957.             for (final Type bound : getImplicitUpperBounds((WildcardType) type)) {
  958.                 if (isAssignable(bound, toGenericArrayType)) {
  959.                     return true;
  960.                 }
  961.             }

  962.             return false;
  963.         }

  964.         if (type instanceof TypeVariable<?>) {
  965.             // probably should remove the following logic and just return false.
  966.             // type variables cannot specify arrays as bounds.
  967.             for (final Type bound : getImplicitBounds((TypeVariable<?>) type)) {
  968.                 if (isAssignable(bound, toGenericArrayType)) {
  969.                     return true;
  970.                 }
  971.             }

  972.             return false;
  973.         }

  974.         if (type instanceof ParameterizedType) {
  975.             // the raw type of a parameterized type is never an array or
  976.             // generic array, otherwise the declaration would look like this:
  977.             // Collection[]< ? extends String > collection;
  978.             return false;
  979.         }

  980.         throw new IllegalStateException("found an unhandled type: " + type);
  981.     }

  982.     /**
  983.      * Tests if the subject type may be implicitly cast to the target parameterized type following the Java generics rules.
  984.      *
  985.      * @param type                the subject type to be assigned to the target type
  986.      * @param toParameterizedType the target parameterized type
  987.      * @param typeVarAssigns      a map with type variables
  988.      * @return {@code true} if {@code type} is assignable to {@code toType}.
  989.      */
  990.     private static boolean isAssignable(final Type type, final ParameterizedType toParameterizedType, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  991.         if (type == null) {
  992.             return true;
  993.         }

  994.         // only a null type can be assigned to null type which
  995.         // would have cause the previous to return true
  996.         if (toParameterizedType == null) {
  997.             return false;
  998.         }

  999.         // cannot cast an array type to a parameterized type.
  1000.         if (type instanceof GenericArrayType) {
  1001.             return false;
  1002.         }

  1003.         // all types are assignable to themselves
  1004.         if (toParameterizedType.equals(type)) {
  1005.             return true;
  1006.         }

  1007.         // get the target type's raw type
  1008.         final Class<?> toClass = getRawType(toParameterizedType);
  1009.         // get the subject type's type arguments including owner type arguments
  1010.         // and supertype arguments up to and including the target class.
  1011.         final Map<TypeVariable<?>, Type> fromTypeVarAssigns = getTypeArguments(type, toClass, null);

  1012.         // null means the two types are not compatible
  1013.         if (fromTypeVarAssigns == null) {
  1014.             return false;
  1015.         }

  1016.         // compatible types, but there's no type arguments. this is equivalent
  1017.         // to comparing Map< ?, ? > to Map, and raw types are always assignable
  1018.         // to parameterized types.
  1019.         if (fromTypeVarAssigns.isEmpty()) {
  1020.             return true;
  1021.         }

  1022.         // get the target type's type arguments including owner type arguments
  1023.         final Map<TypeVariable<?>, Type> toTypeVarAssigns = getTypeArguments(toParameterizedType, toClass, typeVarAssigns);

  1024.         // now to check each type argument
  1025.         for (final TypeVariable<?> var : toTypeVarAssigns.keySet()) {
  1026.             final Type toTypeArg = unrollVariableAssignments(var, toTypeVarAssigns);
  1027.             final Type fromTypeArg = unrollVariableAssignments(var, fromTypeVarAssigns);

  1028.             if (toTypeArg == null && fromTypeArg instanceof Class) {
  1029.                 continue;
  1030.             }

  1031.             // parameters must either be absent from the subject type, within
  1032.             // the bounds of the wildcard type, or be an exact match to the
  1033.             // parameters of the target type.
  1034.             if (fromTypeArg != null && toTypeArg != null && !toTypeArg.equals(fromTypeArg)
  1035.                     && !(toTypeArg instanceof WildcardType && isAssignable(fromTypeArg, toTypeArg, typeVarAssigns))) {
  1036.                 return false;
  1037.             }
  1038.         }
  1039.         return true;
  1040.     }

  1041.     /**
  1042.      * Tests if the subject type may be implicitly cast to the target type following the Java generics rules. If both types are {@link Class} objects, the
  1043.      * method returns the result of {@link ClassUtils#isAssignable(Class, Class)}.
  1044.      *
  1045.      * @param type   the subject type to be assigned to the target type
  1046.      * @param toType the target type
  1047.      * @return {@code true} if {@code type} is assignable to {@code toType}.
  1048.      */
  1049.     public static boolean isAssignable(final Type type, final Type toType) {
  1050.         return isAssignable(type, toType, null);
  1051.     }

  1052.     /**
  1053.      * Tests if the subject type may be implicitly cast to the target type following the Java generics rules.
  1054.      *
  1055.      * @param type           the subject type to be assigned to the target type
  1056.      * @param toType         the target type
  1057.      * @param typeVarAssigns optional map of type variable assignments
  1058.      * @return {@code true} if {@code type} is assignable to {@code toType}.
  1059.      */
  1060.     private static boolean isAssignable(final Type type, final Type toType, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1061.         if (toType == null || toType instanceof Class<?>) {
  1062.             return isAssignable(type, (Class<?>) toType);
  1063.         }

  1064.         if (toType instanceof ParameterizedType) {
  1065.             return isAssignable(type, (ParameterizedType) toType, typeVarAssigns);
  1066.         }

  1067.         if (toType instanceof GenericArrayType) {
  1068.             return isAssignable(type, (GenericArrayType) toType, typeVarAssigns);
  1069.         }

  1070.         if (toType instanceof WildcardType) {
  1071.             return isAssignable(type, (WildcardType) toType, typeVarAssigns);
  1072.         }

  1073.         if (toType instanceof TypeVariable<?>) {
  1074.             return isAssignable(type, (TypeVariable<?>) toType, typeVarAssigns);
  1075.         }

  1076.         throw new IllegalStateException("found an unhandled type: " + toType);
  1077.     }

  1078.     /**
  1079.      * Tests if the subject type may be implicitly cast to the target type variable following the Java generics rules.
  1080.      *
  1081.      * @param type           the subject type to be assigned to the target type
  1082.      * @param toTypeVariable the target type variable
  1083.      * @param typeVarAssigns a map with type variables
  1084.      * @return {@code true} if {@code type} is assignable to {@code toTypeVariable}.
  1085.      */
  1086.     private static boolean isAssignable(final Type type, final TypeVariable<?> toTypeVariable, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1087.         if (type == null) {
  1088.             return true;
  1089.         }

  1090.         // only a null type can be assigned to null type which
  1091.         // would have cause the previous to return true
  1092.         if (toTypeVariable == null) {
  1093.             return false;
  1094.         }

  1095.         // all types are assignable to themselves
  1096.         if (toTypeVariable.equals(type)) {
  1097.             return true;
  1098.         }

  1099.         if (type instanceof TypeVariable<?>) {
  1100.             // a type variable is assignable to another type variable, if
  1101.             // and only if the former is the latter, extends the latter, or
  1102.             // is otherwise a descendant of the latter.
  1103.             final Type[] bounds = getImplicitBounds((TypeVariable<?>) type);

  1104.             for (final Type bound : bounds) {
  1105.                 if (isAssignable(bound, toTypeVariable, typeVarAssigns)) {
  1106.                     return true;
  1107.                 }
  1108.             }
  1109.         }

  1110.         if (type instanceof Class<?> || type instanceof ParameterizedType || type instanceof GenericArrayType || type instanceof WildcardType) {
  1111.             return false;
  1112.         }

  1113.         throw new IllegalStateException("found an unhandled type: " + type);
  1114.     }

  1115.     /**
  1116.      * Tests if the subject type may be implicitly cast to the target wildcard type following the Java generics rules.
  1117.      *
  1118.      * @param type           the subject type to be assigned to the target type
  1119.      * @param toWildcardType the target wildcard type
  1120.      * @param typeVarAssigns a map with type variables
  1121.      * @return {@code true} if {@code type} is assignable to {@code toWildcardType}.
  1122.      */
  1123.     private static boolean isAssignable(final Type type, final WildcardType toWildcardType, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1124.         if (type == null) {
  1125.             return true;
  1126.         }

  1127.         // only a null type can be assigned to null type which
  1128.         // would have cause the previous to return true
  1129.         if (toWildcardType == null) {
  1130.             return false;
  1131.         }

  1132.         // all types are assignable to themselves
  1133.         if (toWildcardType.equals(type)) {
  1134.             return true;
  1135.         }

  1136.         final Type[] toUpperBounds = getImplicitUpperBounds(toWildcardType);
  1137.         final Type[] toLowerBounds = getImplicitLowerBounds(toWildcardType);

  1138.         if (type instanceof WildcardType) {
  1139.             final WildcardType wildcardType = (WildcardType) type;
  1140.             final Type[] upperBounds = getImplicitUpperBounds(wildcardType);
  1141.             final Type[] lowerBounds = getImplicitLowerBounds(wildcardType);

  1142.             for (Type toBound : toUpperBounds) {
  1143.                 // if there are assignments for unresolved type variables,
  1144.                 // now's the time to substitute them.
  1145.                 toBound = substituteTypeVariables(toBound, typeVarAssigns);

  1146.                 // each upper bound of the subject type has to be assignable to
  1147.                 // each
  1148.                 // upper bound of the target type
  1149.                 for (final Type bound : upperBounds) {
  1150.                     if (!isAssignable(bound, toBound, typeVarAssigns)) {
  1151.                         return false;
  1152.                     }
  1153.                 }
  1154.             }

  1155.             for (Type toBound : toLowerBounds) {
  1156.                 // if there are assignments for unresolved type variables,
  1157.                 // now's the time to substitute them.
  1158.                 toBound = substituteTypeVariables(toBound, typeVarAssigns);

  1159.                 // each lower bound of the target type has to be assignable to
  1160.                 // each
  1161.                 // lower bound of the subject type
  1162.                 for (final Type bound : lowerBounds) {
  1163.                     if (!isAssignable(toBound, bound, typeVarAssigns)) {
  1164.                         return false;
  1165.                     }
  1166.                 }
  1167.             }
  1168.             return true;
  1169.         }

  1170.         for (final Type toBound : toUpperBounds) {
  1171.             // if there are assignments for unresolved type variables,
  1172.             // now's the time to substitute them.
  1173.             if (!isAssignable(type, substituteTypeVariables(toBound, typeVarAssigns), typeVarAssigns)) {
  1174.                 return false;
  1175.             }
  1176.         }

  1177.         for (final Type toBound : toLowerBounds) {
  1178.             // if there are assignments for unresolved type variables,
  1179.             // now's the time to substitute them.
  1180.             if (!isAssignable(substituteTypeVariables(toBound, typeVarAssigns), type, typeVarAssigns)) {
  1181.                 return false;
  1182.             }
  1183.         }
  1184.         return true;
  1185.     }

  1186.     /**
  1187.      * Tests whether the class contains a cyclical reference in the qualified name of a class. If any of the type parameters of A class is extending X class
  1188.      * which is in scope of A class, then it forms cycle.
  1189.      *
  1190.      * @param cls the class to test.
  1191.      * @return whether the class contains a cyclical reference.
  1192.      */
  1193.     private static boolean isCyclical(final Class<?> cls) {
  1194.         for (final TypeVariable<?> typeParameter : cls.getTypeParameters()) {
  1195.             for (final AnnotatedType annotatedBound : typeParameter.getAnnotatedBounds()) {
  1196.                 if (annotatedBound.getType().getTypeName().contains(cls.getName())) {
  1197.                     return true;
  1198.                 }
  1199.             }
  1200.         }
  1201.         return false;
  1202.     }

  1203.     /**
  1204.      * Tests if the given value can be assigned to the target type following the Java generics rules.
  1205.      *
  1206.      * @param value the value to be checked
  1207.      * @param type  the target type
  1208.      * @return {@code true} if {@code value} is an instance of {@code type}.
  1209.      */
  1210.     public static boolean isInstance(final Object value, final Type type) {
  1211.         if (type == null) {
  1212.             return false;
  1213.         }

  1214.         return value == null ? !(type instanceof Class<?>) || !((Class<?>) type).isPrimitive() : isAssignable(value.getClass(), type, null);
  1215.     }

  1216.     /**
  1217.      * Maps type variables.
  1218.      *
  1219.      * @param <T>               the generic type of the class in question
  1220.      * @param cls               the class in question
  1221.      * @param parameterizedType the parameterized type
  1222.      * @param typeVarAssigns    the map to be filled
  1223.      */
  1224.     private static <T> void mapTypeVariablesToArguments(final Class<T> cls, final ParameterizedType parameterizedType,
  1225.             final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1226.         // capture the type variables from the owner type that have assignments
  1227.         final Type ownerType = parameterizedType.getOwnerType();

  1228.         if (ownerType instanceof ParameterizedType) {
  1229.             // recursion to make sure the owner's owner type gets processed
  1230.             mapTypeVariablesToArguments(cls, (ParameterizedType) ownerType, typeVarAssigns);
  1231.         }

  1232.         // parameterizedType is a generic interface/class (or it's in the owner
  1233.         // hierarchy of said interface/class) implemented/extended by the class
  1234.         // cls. Find out which type variables of cls are type arguments of
  1235.         // parameterizedType:
  1236.         final Type[] typeArgs = parameterizedType.getActualTypeArguments();

  1237.         // of the cls's type variables that are arguments of parameterizedType,
  1238.         // find out which ones can be determined from the super type's arguments
  1239.         final TypeVariable<?>[] typeVars = getRawType(parameterizedType).getTypeParameters();

  1240.         // use List view of type parameters of cls so the contains() method can be used:
  1241.         final List<TypeVariable<Class<T>>> typeVarList = Arrays.asList(cls.getTypeParameters());

  1242.         for (int i = 0; i < typeArgs.length; i++) {
  1243.             final TypeVariable<?> typeVar = typeVars[i];
  1244.             final Type typeArg = typeArgs[i];

  1245.             // argument of parameterizedType is a type variable of cls
  1246.             if (typeVarList.contains(typeArg)
  1247.                     // type variable of parameterizedType has an assignment in
  1248.                     // the super type.
  1249.                     && typeVarAssigns.containsKey(typeVar)) {
  1250.                 // map the assignment to the cls's type variable
  1251.                 typeVarAssigns.put((TypeVariable<?>) typeArg, typeVarAssigns.get(typeVar));
  1252.             }
  1253.         }
  1254.     }

  1255.     /**
  1256.      * Strips out the redundant upper bound types in type variable types and wildcard types (or it would with wildcard types if multiple upper bounds were
  1257.      * allowed).
  1258.      *
  1259.      * <p>
  1260.      * Example, with the variable type declaration:
  1261.      * </p>
  1262.      *
  1263.      * <pre>{@code
  1264.      * <K extends java.util.Collection<String> & java.util.List<String>>
  1265.      * }</pre>
  1266.      *
  1267.      * <p>
  1268.      * since {@link List} is a subinterface of {@link Collection}, this method will return the bounds as if the declaration had been:
  1269.      * </p>
  1270.      *
  1271.      * <pre>{@code
  1272.      * <K extends java.util.List<String>>
  1273.      * }</pre>
  1274.      *
  1275.      * @param bounds an array of types representing the upper bounds of either {@link WildcardType} or {@link TypeVariable}, not {@code null}.
  1276.      * @return an array containing the values from {@code bounds} minus the redundant types.
  1277.      * @throws NullPointerException if {@code bounds} is {@code null}
  1278.      */
  1279.     public static Type[] normalizeUpperBounds(final Type[] bounds) {
  1280.         Objects.requireNonNull(bounds, "bounds");
  1281.         // don't bother if there's only one (or none) type
  1282.         if (bounds.length < 2) {
  1283.             return bounds;
  1284.         }

  1285.         final Set<Type> types = new HashSet<>(bounds.length);

  1286.         for (final Type type1 : bounds) {
  1287.             boolean subtypeFound = false;

  1288.             for (final Type type2 : bounds) {
  1289.                 if (type1 != type2 && isAssignable(type2, type1, null)) {
  1290.                     subtypeFound = true;
  1291.                     break;
  1292.                 }
  1293.             }

  1294.             if (!subtypeFound) {
  1295.                 types.add(type1);
  1296.             }
  1297.         }

  1298.         return types.toArray(ArrayUtils.EMPTY_TYPE_ARRAY);
  1299.     }

  1300.     /**
  1301.      * Creates a parameterized type instance.
  1302.      *
  1303.      * @param rawClass        the raw class to create a parameterized type instance for
  1304.      * @param typeVariableMap the map used for parameterization
  1305.      * @return {@link ParameterizedType}
  1306.      * @throws NullPointerException if either {@code rawClass} or {@code typeVariableMap} is {@code null}
  1307.      * @since 3.2
  1308.      */
  1309.     public static final ParameterizedType parameterize(final Class<?> rawClass, final Map<TypeVariable<?>, Type> typeVariableMap) {
  1310.         Objects.requireNonNull(rawClass, "rawClass");
  1311.         Objects.requireNonNull(typeVariableMap, "typeVariableMap");
  1312.         return parameterizeWithOwner(null, rawClass, extractTypeArgumentsFrom(typeVariableMap, rawClass.getTypeParameters()));
  1313.     }

  1314.     /**
  1315.      * Creates a parameterized type instance.
  1316.      *
  1317.      * @param rawClass      the raw class to create a parameterized type instance for
  1318.      * @param typeArguments the types used for parameterization
  1319.      * @return {@link ParameterizedType}
  1320.      * @throws NullPointerException if {@code rawClass} is {@code null}
  1321.      * @since 3.2
  1322.      */
  1323.     public static final ParameterizedType parameterize(final Class<?> rawClass, final Type... typeArguments) {
  1324.         return parameterizeWithOwner(null, rawClass, typeArguments);
  1325.     }

  1326.     /**
  1327.      * Formats a {@link ParameterizedType} as a {@link String}.
  1328.      *
  1329.      * @param parameterizedType {@link ParameterizedType} to format
  1330.      * @return String
  1331.      */
  1332.     private static String parameterizedTypeToString(final ParameterizedType parameterizedType) {
  1333.         final StringBuilder builder = new StringBuilder();
  1334.         final Type useOwner = parameterizedType.getOwnerType();
  1335.         final Class<?> raw = (Class<?>) parameterizedType.getRawType();
  1336.         if (useOwner == null) {
  1337.             builder.append(raw.getName());
  1338.         } else {
  1339.             if (useOwner instanceof Class<?>) {
  1340.                 builder.append(((Class<?>) useOwner).getName());
  1341.             } else {
  1342.                 builder.append(useOwner);
  1343.             }
  1344.             builder.append('.').append(raw.getSimpleName());
  1345.         }
  1346.         final int[] recursiveTypeIndexes = findRecursiveTypes(parameterizedType);
  1347.         if (recursiveTypeIndexes.length > 0) {
  1348.             appendRecursiveTypes(builder, recursiveTypeIndexes, parameterizedType.getActualTypeArguments());
  1349.         } else {
  1350.             GT_JOINER.join(builder, parameterizedType.getActualTypeArguments());
  1351.         }
  1352.         return builder.toString();
  1353.     }

  1354.     /**
  1355.      * Creates a parameterized type instance.
  1356.      *
  1357.      * @param owner           the owning type
  1358.      * @param rawClass        the raw class to create a parameterized type instance for
  1359.      * @param typeVariableMap the map used for parameterization
  1360.      * @return {@link ParameterizedType}
  1361.      * @throws NullPointerException if either {@code rawClass} or {@code typeVariableMap} is {@code null}
  1362.      * @since 3.2
  1363.      */
  1364.     public static final ParameterizedType parameterizeWithOwner(final Type owner, final Class<?> rawClass, final Map<TypeVariable<?>, Type> typeVariableMap) {
  1365.         Objects.requireNonNull(rawClass, "rawClass");
  1366.         Objects.requireNonNull(typeVariableMap, "typeVariableMap");
  1367.         return parameterizeWithOwner(owner, rawClass, extractTypeArgumentsFrom(typeVariableMap, rawClass.getTypeParameters()));
  1368.     }

  1369.     /**
  1370.      * Creates a parameterized type instance.
  1371.      *
  1372.      * @param owner         the owning type
  1373.      * @param rawClass      the raw class to create a parameterized type instance for
  1374.      * @param typeArguments the types used for parameterization
  1375.      *
  1376.      * @return {@link ParameterizedType}
  1377.      * @throws NullPointerException if {@code rawClass} is {@code null}
  1378.      * @since 3.2
  1379.      */
  1380.     public static final ParameterizedType parameterizeWithOwner(final Type owner, final Class<?> rawClass, final Type... typeArguments) {
  1381.         Objects.requireNonNull(rawClass, "rawClass");
  1382.         final Type useOwner;
  1383.         if (rawClass.getEnclosingClass() == null) {
  1384.             Validate.isTrue(owner == null, "no owner allowed for top-level %s", rawClass);
  1385.             useOwner = null;
  1386.         } else if (owner == null) {
  1387.             useOwner = rawClass.getEnclosingClass();
  1388.         } else {
  1389.             Validate.isTrue(isAssignable(owner, rawClass.getEnclosingClass()), "%s is invalid owner type for parameterized %s", owner, rawClass);
  1390.             useOwner = owner;
  1391.         }
  1392.         Validate.noNullElements(typeArguments, "null type argument at index %s");
  1393.         Validate.isTrue(rawClass.getTypeParameters().length == typeArguments.length, "invalid number of type parameters specified: expected %d, got %d",
  1394.                 rawClass.getTypeParameters().length, typeArguments.length);

  1395.         return new ParameterizedTypeImpl(rawClass, useOwner, typeArguments);
  1396.     }

  1397.     /**
  1398.      * Finds the mapping for {@code type} in {@code typeVarAssigns}.
  1399.      *
  1400.      * @param type           the type to be replaced
  1401.      * @param typeVarAssigns the map with type variables
  1402.      * @return the replaced type
  1403.      * @throws IllegalArgumentException if the type cannot be substituted
  1404.      */
  1405.     private static Type substituteTypeVariables(final Type type, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1406.         if (type instanceof TypeVariable<?> && typeVarAssigns != null) {
  1407.             final Type replacementType = typeVarAssigns.get(type);

  1408.             if (replacementType == null) {
  1409.                 throw new IllegalArgumentException("missing assignment type for type variable " + type);
  1410.             }
  1411.             return replacementType;
  1412.         }
  1413.         return type;
  1414.     }

  1415.     /**
  1416.      * Formats a {@link TypeVariable} including its {@link GenericDeclaration}.
  1417.      *
  1418.      * @param typeVariable the type variable to create a String representation for, not {@code null}
  1419.      * @return String
  1420.      * @throws NullPointerException if {@code typeVariable} is {@code null}
  1421.      * @since 3.2
  1422.      */
  1423.     public static String toLongString(final TypeVariable<?> typeVariable) {
  1424.         Objects.requireNonNull(typeVariable, "typeVariable");
  1425.         final StringBuilder buf = new StringBuilder();
  1426.         final GenericDeclaration d = typeVariable.getGenericDeclaration();
  1427.         if (d instanceof Class<?>) {
  1428.             Class<?> c = (Class<?>) d;
  1429.             while (true) {
  1430.                 if (c.getEnclosingClass() == null) {
  1431.                     buf.insert(0, c.getName());
  1432.                     break;
  1433.                 }
  1434.                 buf.insert(0, c.getSimpleName()).insert(0, '.');
  1435.                 c = c.getEnclosingClass();
  1436.             }
  1437.         } else if (d instanceof Type) { // not possible as of now
  1438.             buf.append(toString((Type) d));
  1439.         } else {
  1440.             buf.append(d);
  1441.         }
  1442.         return buf.append(':').append(typeVariableToString(typeVariable)).toString();
  1443.     }

  1444.     /**
  1445.      * Formats a given type as a Java-esque String.
  1446.      *
  1447.      * @param type the type to create a String representation for, not {@code null}
  1448.      * @return String
  1449.      * @throws NullPointerException if {@code type} is {@code null}
  1450.      * @since 3.2
  1451.      */
  1452.     public static String toString(final Type type) {
  1453.         Objects.requireNonNull(type, "type");
  1454.         if (type instanceof Class<?>) {
  1455.             return classToString((Class<?>) type);
  1456.         }
  1457.         if (type instanceof ParameterizedType) {
  1458.             return parameterizedTypeToString((ParameterizedType) type);
  1459.         }
  1460.         if (type instanceof WildcardType) {
  1461.             return wildcardTypeToString((WildcardType) type);
  1462.         }
  1463.         if (type instanceof TypeVariable<?>) {
  1464.             return typeVariableToString((TypeVariable<?>) type);
  1465.         }
  1466.         if (type instanceof GenericArrayType) {
  1467.             return genericArrayTypeToString((GenericArrayType) type);
  1468.         }
  1469.         throw new IllegalArgumentException(ObjectUtils.identityToString(type));
  1470.     }

  1471.     /**
  1472.      * Determines whether or not specified types satisfy the bounds of their mapped type variables. When a type parameter extends another (such as
  1473.      * {@code <T, S extends T>}), uses another as a type parameter (such as {@code <T, S extends Comparable>>}), or otherwise depends on another type variable
  1474.      * to be specified, the dependencies must be included in {@code typeVarAssigns}.
  1475.      *
  1476.      * @param typeVariableMap specifies the potential types to be assigned to the type variables, not {@code null}.
  1477.      * @return whether or not the types can be assigned to their respective type variables.
  1478.      * @throws NullPointerException if {@code typeVariableMap} is {@code null}
  1479.      */
  1480.     public static boolean typesSatisfyVariables(final Map<TypeVariable<?>, Type> typeVariableMap) {
  1481.         Objects.requireNonNull(typeVariableMap, "typeVariableMap");
  1482.         // all types must be assignable to all the bounds of their mapped
  1483.         // type variable.
  1484.         for (final Map.Entry<TypeVariable<?>, Type> entry : typeVariableMap.entrySet()) {
  1485.             final TypeVariable<?> typeVar = entry.getKey();
  1486.             final Type type = entry.getValue();

  1487.             for (final Type bound : getImplicitBounds(typeVar)) {
  1488.                 if (!isAssignable(type, substituteTypeVariables(bound, typeVariableMap), typeVariableMap)) {
  1489.                     return false;
  1490.                 }
  1491.             }
  1492.         }
  1493.         return true;
  1494.     }

  1495.     /**
  1496.      * Formats a {@link TypeVariable} as a {@link String}.
  1497.      *
  1498.      * @param typeVariable {@link TypeVariable} to format
  1499.      * @return String
  1500.      */
  1501.     private static String typeVariableToString(final TypeVariable<?> typeVariable) {
  1502.         final StringBuilder builder = new StringBuilder(typeVariable.getName());
  1503.         final Type[] bounds = typeVariable.getBounds();
  1504.         if (bounds.length > 0 && !(bounds.length == 1 && Object.class.equals(bounds[0]))) {
  1505.             builder.append(" extends ");
  1506.             AMP_JOINER.join(builder, typeVariable.getBounds());
  1507.         }
  1508.         return builder.toString();
  1509.     }

  1510.     /**
  1511.      * Unrolls variables in a type bounds array.
  1512.      *
  1513.      * @param typeArguments assignments {@link Map}
  1514.      * @param bounds        in which to expand variables
  1515.      * @return {@code bounds} with any variables reassigned
  1516.      */
  1517.     private static Type[] unrollBounds(final Map<TypeVariable<?>, Type> typeArguments, final Type[] bounds) {
  1518.         Type[] result = bounds;
  1519.         int i = 0;
  1520.         for (; i < result.length; i++) {
  1521.             final Type unrolled = unrollVariables(typeArguments, result[i]);
  1522.             if (unrolled == null) {
  1523.                 result = ArrayUtils.remove(result, i--);
  1524.             } else {
  1525.                 result[i] = unrolled;
  1526.             }
  1527.         }
  1528.         return result;
  1529.     }

  1530.     /**
  1531.      * Looks up {@code typeVariable} in {@code typeVarAssigns} <em>transitively</em>, i.e. keep looking until the value found is <em>not</em> a type variable.
  1532.      *
  1533.      * @param typeVariable   the type variable to look up
  1534.      * @param typeVarAssigns the map used for the look-up
  1535.      * @return Type or {@code null} if some variable was not in the map
  1536.      */
  1537.     private static Type unrollVariableAssignments(TypeVariable<?> typeVariable, final Map<TypeVariable<?>, Type> typeVarAssigns) {
  1538.         Type result;
  1539.         do {
  1540.             result = typeVarAssigns.get(typeVariable);
  1541.             if (!(result instanceof TypeVariable<?>) || result.equals(typeVariable)) {
  1542.                 break;
  1543.             }
  1544.             typeVariable = (TypeVariable<?>) result;
  1545.         } while (true);
  1546.         return result;
  1547.     }

  1548.     /**
  1549.      * Gets a type representing {@code type} with variable assignments "unrolled."
  1550.      *
  1551.      * @param typeArguments as from {@link TypeUtils#getTypeArguments(Type, Class)}
  1552.      * @param type          the type to unroll variable assignments for
  1553.      * @return Type
  1554.      * @since 3.2
  1555.      */
  1556.     public static Type unrollVariables(Map<TypeVariable<?>, Type> typeArguments, final Type type) {
  1557.         if (typeArguments == null) {
  1558.             typeArguments = Collections.emptyMap();
  1559.         }
  1560.         if (containsTypeVariables(type)) {
  1561.             if (type instanceof TypeVariable<?>) {
  1562.                 return unrollVariables(typeArguments, typeArguments.get(type));
  1563.             }
  1564.             if (type instanceof ParameterizedType) {
  1565.                 final ParameterizedType p = (ParameterizedType) type;
  1566.                 final Map<TypeVariable<?>, Type> parameterizedTypeArguments;
  1567.                 if (p.getOwnerType() == null) {
  1568.                     parameterizedTypeArguments = typeArguments;
  1569.                 } else {
  1570.                     parameterizedTypeArguments = new HashMap<>(typeArguments);
  1571.                     parameterizedTypeArguments.putAll(getTypeArguments(p));
  1572.                 }
  1573.                 final Type[] args = p.getActualTypeArguments();
  1574.                 for (int i = 0; i < args.length; i++) {
  1575.                     final Type unrolled = unrollVariables(parameterizedTypeArguments, args[i]);
  1576.                     if (unrolled != null) {
  1577.                         args[i] = unrolled;
  1578.                     }
  1579.                 }
  1580.                 return parameterizeWithOwner(p.getOwnerType(), (Class<?>) p.getRawType(), args);
  1581.             }
  1582.             if (type instanceof WildcardType) {
  1583.                 final WildcardType wild = (WildcardType) type;
  1584.                 return wildcardType().withUpperBounds(unrollBounds(typeArguments, wild.getUpperBounds()))
  1585.                         .withLowerBounds(unrollBounds(typeArguments, wild.getLowerBounds())).build();
  1586.             }
  1587.         }
  1588.         return type;
  1589.     }

  1590.     /**
  1591.      * Gets a {@link WildcardTypeBuilder}.
  1592.      *
  1593.      * @return {@link WildcardTypeBuilder}
  1594.      * @since 3.2
  1595.      */
  1596.     public static WildcardTypeBuilder wildcardType() {
  1597.         return new WildcardTypeBuilder();
  1598.     }

  1599.     /**
  1600.      * Formats a {@link WildcardType} as a {@link String}.
  1601.      *
  1602.      * @param wildcardType {@link WildcardType} to format
  1603.      * @return String
  1604.      */
  1605.     private static String wildcardTypeToString(final WildcardType wildcardType) {
  1606.         final StringBuilder builder = new StringBuilder().append('?');
  1607.         final Type[] lowerBounds = wildcardType.getLowerBounds();
  1608.         final Type[] upperBounds = wildcardType.getUpperBounds();
  1609.         if (lowerBounds.length > 1 || lowerBounds.length == 1 && lowerBounds[0] != null) {
  1610.             AMP_JOINER.join(builder.append(" super "), lowerBounds);
  1611.         } else if (upperBounds.length > 1 || upperBounds.length == 1 && !Object.class.equals(upperBounds[0])) {
  1612.             AMP_JOINER.join(builder.append(" extends "), upperBounds);
  1613.         }
  1614.         return builder.toString();
  1615.     }

  1616.     /**
  1617.      * Wraps the specified {@link Class} in a {@link Typed} wrapper.
  1618.      *
  1619.      * @param <T>  generic type
  1620.      * @param type to wrap
  1621.      * @return {@code Typed<T>}
  1622.      * @since 3.2
  1623.      */
  1624.     public static <T> Typed<T> wrap(final Class<T> type) {
  1625.         return wrap((Type) type);
  1626.     }

  1627.     /**
  1628.      * Wraps the specified {@link Type} in a {@link Typed} wrapper.
  1629.      *
  1630.      * @param <T>  inferred generic type
  1631.      * @param type to wrap
  1632.      * @return {@code Typed<T>}
  1633.      * @since 3.2
  1634.      */
  1635.     public static <T> Typed<T> wrap(final Type type) {
  1636.         return () -> type;
  1637.     }

  1638.     /**
  1639.      * {@link TypeUtils} instances should NOT be constructed in standard programming. Instead, the class should be used as
  1640.      * {@code TypeUtils.isAssignable(cls, toClass)}.
  1641.      * <p>
  1642.      * This constructor is public to permit tools that require a JavaBean instance to operate.
  1643.      * </p>
  1644.      *
  1645.      * @deprecated TODO Make private in 4.0.
  1646.      */
  1647.     @Deprecated
  1648.     public TypeUtils() {
  1649.         // empty
  1650.     }

  1651. }