MemberUtils.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.AccessibleObject;
  19. import java.lang.reflect.Constructor;
  20. import java.lang.reflect.Member;
  21. import java.lang.reflect.Method;
  22. import java.lang.reflect.Modifier;

  23. import org.apache.commons.lang3.ClassUtils;

  24. /**
  25.  * Contains common code for working with {@link java.lang.reflect.Method Methods}/{@link java.lang.reflect.Constructor Constructors},
  26.  * extracted and refactored from {@link MethodUtils} when it was imported from Commons BeanUtils.
  27.  *
  28.  * @since 2.5
  29.  */
  30. final class MemberUtils {
  31.     // TODO extract an interface to implement compareParameterSets(...)?

  32.     /**
  33.      *  A class providing a subset of the API of java.lang.reflect.Executable in Java 1.8,
  34.      * providing a common representation for function signatures for Constructors and Methods.
  35.      */
  36.     private static final class Executable {
  37.       private static Executable of(final Constructor<?> constructor) {
  38.           return new Executable(constructor);
  39.       }
  40.       private static Executable of(final Method method) {
  41.           return new Executable(method);
  42.       }

  43.       private final Class<?>[] parameterTypes;

  44.       private final boolean  isVarArgs;

  45.       private Executable(final Constructor<?> constructor) {
  46.         parameterTypes = constructor.getParameterTypes();
  47.         isVarArgs = constructor.isVarArgs();
  48.       }

  49.       private Executable(final Method method) {
  50.         parameterTypes = method.getParameterTypes();
  51.         isVarArgs = method.isVarArgs();
  52.       }

  53.       public Class<?>[] getParameterTypes() {
  54.           return parameterTypes;
  55.       }

  56.       public boolean isVarArgs() {
  57.           return isVarArgs;
  58.       }
  59.     }

  60.     private static final int ACCESS_TEST = Modifier.PUBLIC | Modifier.PROTECTED | Modifier.PRIVATE;

  61.     /** Array of primitive number types ordered by "promotability" */
  62.     private static final Class<?>[] ORDERED_PRIMITIVE_TYPES = { Byte.TYPE, Short.TYPE,
  63.             Character.TYPE, Integer.TYPE, Long.TYPE, Float.TYPE, Double.TYPE };

  64.     /**
  65.      * Compares the relative fitness of two Constructors in terms of how well they
  66.      * match a set of runtime parameter types, such that a list ordered
  67.      * by the results of the comparison would return the best match first
  68.      * (least).
  69.      *
  70.      * @param left the "left" Constructor
  71.      * @param right the "right" Constructor
  72.      * @param actual the runtime parameter types to match against
  73.      * {@code left}/{@code right}
  74.      * @return int consistent with {@code compare} semantics
  75.      * @since 3.5
  76.      */
  77.     static int compareConstructorFit(final Constructor<?> left, final Constructor<?> right, final Class<?>[] actual) {
  78.       return compareParameterTypes(Executable.of(left), Executable.of(right), actual);
  79.     }

  80.     /**
  81.      * Compares the relative fitness of two Methods in terms of how well they
  82.      * match a set of runtime parameter types, such that a list ordered
  83.      * by the results of the comparison would return the best match first
  84.      * (least).
  85.      *
  86.      * @param left the "left" Method
  87.      * @param right the "right" Method
  88.      * @param actual the runtime parameter types to match against
  89.      * {@code left}/{@code right}
  90.      * @return int consistent with {@code compare} semantics
  91.      * @since 3.5
  92.      */
  93.     static int compareMethodFit(final Method left, final Method right, final Class<?>[] actual) {
  94.       return compareParameterTypes(Executable.of(left), Executable.of(right), actual);
  95.     }

  96.     /**
  97.      * Compares the relative fitness of two Executables in terms of how well they
  98.      * match a set of runtime parameter types, such that a list ordered
  99.      * by the results of the comparison would return the best match first
  100.      * (least).
  101.      *
  102.      * @param left the "left" Executable
  103.      * @param right the "right" Executable
  104.      * @param actual the runtime parameter types to match against
  105.      * {@code left}/{@code right}
  106.      * @return int consistent with {@code compare} semantics
  107.      */
  108.     private static int compareParameterTypes(final Executable left, final Executable right, final Class<?>[] actual) {
  109.         final float leftCost = getTotalTransformationCost(actual, left);
  110.         final float rightCost = getTotalTransformationCost(actual, right);
  111.         return Float.compare(leftCost, rightCost);
  112.     }

  113.     /**
  114.      * Gets the number of steps needed to turn the source class into
  115.      * the destination class. This represents the number of steps in the object
  116.      * hierarchy graph.
  117.      * @param srcClass The source class
  118.      * @param destClass The destination class
  119.      * @return The cost of transforming an object
  120.      */
  121.     private static float getObjectTransformationCost(Class<?> srcClass, final Class<?> destClass) {
  122.         if (destClass.isPrimitive()) {
  123.             return getPrimitivePromotionCost(srcClass, destClass);
  124.         }
  125.         float cost = 0.0f;
  126.         while (srcClass != null && !destClass.equals(srcClass)) {
  127.             if (destClass.isInterface() && ClassUtils.isAssignable(srcClass, destClass)) {
  128.                 // slight penalty for interface match.
  129.                 // we still want an exact match to override an interface match,
  130.                 // but
  131.                 // an interface match should override anything where we have to
  132.                 // get a superclass.
  133.                 cost += 0.25f;
  134.                 break;
  135.             }
  136.             cost++;
  137.             srcClass = srcClass.getSuperclass();
  138.         }
  139.         /*
  140.          * If the destination class is null, we've traveled all the way up to
  141.          * an Object match. We'll penalize this by adding 1.5 to the cost.
  142.          */
  143.         if (srcClass == null) {
  144.             cost += 1.5f;
  145.         }
  146.         return cost;
  147.     }

  148.     /**
  149.      * Gets the number of steps required to promote a primitive number to another
  150.      * type.
  151.      * @param srcClass the (primitive) source class
  152.      * @param destClass the (primitive) destination class
  153.      * @return The cost of promoting the primitive
  154.      */
  155.     private static float getPrimitivePromotionCost(final Class<?> srcClass, final Class<?> destClass) {
  156.         if (srcClass == null) {
  157.             return 1.5f;
  158.         }
  159.         float cost = 0.0f;
  160.         Class<?> cls = srcClass;
  161.         if (!cls.isPrimitive()) {
  162.             // slight unwrapping penalty
  163.             cost += 0.1f;
  164.             cls = ClassUtils.wrapperToPrimitive(cls);
  165.         }
  166.         for (int i = 0; cls != destClass && i < ORDERED_PRIMITIVE_TYPES.length; i++) {
  167.             if (cls == ORDERED_PRIMITIVE_TYPES[i]) {
  168.                 cost += 0.1f;
  169.                 if (i < ORDERED_PRIMITIVE_TYPES.length - 1) {
  170.                     cls = ORDERED_PRIMITIVE_TYPES[i + 1];
  171.                 }
  172.             }
  173.         }
  174.         return cost;
  175.     }

  176.     /**
  177.      * Returns the sum of the object transformation cost for each class in the
  178.      * source argument list.
  179.      * @param srcArgs The source arguments
  180.      * @param executable The executable to calculate transformation costs for
  181.      * @return The total transformation cost
  182.      */
  183.     private static float getTotalTransformationCost(final Class<?>[] srcArgs, final Executable executable) {
  184.         final Class<?>[] destArgs = executable.getParameterTypes();
  185.         final boolean isVarArgs = executable.isVarArgs();

  186.         // "source" and "destination" are the actual and declared args respectively.
  187.         float totalCost = 0.0f;
  188.         final long normalArgsLen = isVarArgs ? destArgs.length - 1 : destArgs.length;
  189.         if (srcArgs.length < normalArgsLen) {
  190.             return Float.MAX_VALUE;
  191.         }
  192.         for (int i = 0; i < normalArgsLen; i++) {
  193.             totalCost += getObjectTransformationCost(srcArgs[i], destArgs[i]);
  194.         }
  195.         if (isVarArgs) {
  196.             // When isVarArgs is true, srcArgs and dstArgs may differ in length.
  197.             // There are two special cases to consider:
  198.             final boolean noVarArgsPassed = srcArgs.length < destArgs.length;
  199.             final boolean explicitArrayForVarargs = srcArgs.length == destArgs.length && srcArgs[srcArgs.length - 1] != null
  200.                 && srcArgs[srcArgs.length - 1].isArray();

  201.             final float varArgsCost = 0.001f;
  202.             final Class<?> destClass = destArgs[destArgs.length - 1].getComponentType();
  203.             if (noVarArgsPassed) {
  204.                 // When no varargs passed, the best match is the most generic matching type, not the most specific.
  205.                 totalCost += getObjectTransformationCost(destClass, Object.class) + varArgsCost;
  206.             } else if (explicitArrayForVarargs) {
  207.                 final Class<?> sourceClass = srcArgs[srcArgs.length - 1].getComponentType();
  208.                 totalCost += getObjectTransformationCost(sourceClass, destClass) + varArgsCost;
  209.             } else {
  210.                 // This is typical varargs case.
  211.                 for (int i = destArgs.length - 1; i < srcArgs.length; i++) {
  212.                     final Class<?> srcClass = srcArgs[i];
  213.                     totalCost += getObjectTransformationCost(srcClass, destClass) + varArgsCost;
  214.                 }
  215.             }
  216.         }
  217.         return totalCost;
  218.     }

  219.     /**
  220.      * Tests whether a {@link Member} is accessible.
  221.      *
  222.      * @param member Member to test, may be null.
  223.      * @return {@code true} if {@code m} is accessible
  224.      */
  225.     static boolean isAccessible(final Member member) {
  226.         return isPublic(member) && !member.isSynthetic();
  227.     }

  228.     static boolean isMatchingConstructor(final Constructor<?> method, final Class<?>[] parameterTypes) {
  229.         return isMatchingExecutable(Executable.of(method), parameterTypes);
  230.     }

  231.     private static boolean isMatchingExecutable(final Executable method, final Class<?>[] parameterTypes) {
  232.         final Class<?>[] methodParameterTypes = method.getParameterTypes();
  233.         if (ClassUtils.isAssignable(parameterTypes, methodParameterTypes, true)) {
  234.             return true;
  235.         }
  236.         if (method.isVarArgs()) {
  237.             int i;
  238.             for (i = 0; i < methodParameterTypes.length - 1 && i < parameterTypes.length; i++) {
  239.                 if (!ClassUtils.isAssignable(parameterTypes[i], methodParameterTypes[i], true)) {
  240.                     return false;
  241.                 }
  242.             }
  243.             final Class<?> varArgParameterType = methodParameterTypes[methodParameterTypes.length - 1].getComponentType();
  244.             for (; i < parameterTypes.length; i++) {
  245.                 if (!ClassUtils.isAssignable(parameterTypes[i], varArgParameterType, true)) {
  246.                     return false;
  247.                 }
  248.             }
  249.             return true;
  250.         }
  251.         return false;
  252.     }

  253.     static boolean isMatchingMethod(final Method method, final Class<?>[] parameterTypes) {
  254.       return isMatchingExecutable(Executable.of(method), parameterTypes);
  255.     }

  256.     /**
  257.      * Tests whether a given set of modifiers implies package access.
  258.      *
  259.      * @param modifiers to test
  260.      * @return {@code true} unless {@code package}/{@code protected}/{@code private} modifier detected
  261.      */
  262.     static boolean isPackageAccess(final int modifiers) {
  263.         return (modifiers & ACCESS_TEST) == 0;
  264.     }

  265.     /**
  266.      * Tests whether a {@link Member} is public.
  267.      *
  268.      * @param member Member to test, may be null.
  269.      * @return {@code true} if {@code m} is public
  270.      */
  271.     static boolean isPublic(final Member member) {
  272.         return member != null && Modifier.isPublic(member.getModifiers());
  273.     }

  274.     /**
  275.      * Tests whether a {@link Member} is static.
  276.      *
  277.      * @param member Member to test, may be null.
  278.      * @return {@code true} if {@code m} is static
  279.      */
  280.     static boolean isStatic(final Member member) {
  281.         return member != null && Modifier.isStatic(member.getModifiers());
  282.     }

  283.     /**
  284.      * Default access superclass workaround.
  285.      * <p>
  286.      * When a {@code public} class has a default access superclass with {@code public} members,
  287.      * these members are accessible. Calling them from compiled code works fine.
  288.      * Unfortunately, on some JVMs, using reflection to invoke these members
  289.      * seems to (wrongly) prevent access even when the modifier is {@code public}.
  290.      * Calling {@code setAccessible(true)} solves the problem but will only work from
  291.      * sufficiently privileged code. Better workarounds would be gratefully
  292.      * accepted.
  293.      * </p>
  294.      *
  295.      * @param obj the AccessibleObject to set as accessible, may be null.
  296.      * @return a boolean indicating whether the accessibility of the object was set to true.
  297.      */
  298.     static <T extends AccessibleObject> T setAccessibleWorkaround(final T obj) {
  299.         if (obj == null || obj.isAccessible()) {
  300.             return obj;
  301.         }
  302.         final Member m = (Member) obj;
  303.         if (!obj.isAccessible() && isPublic(m) && isPackageAccess(m.getDeclaringClass().getModifiers())) {
  304.             try {
  305.                 obj.setAccessible(true);
  306.                 return obj;
  307.             } catch (final SecurityException ignored) {
  308.                 // ignore in favor of subsequent IllegalAccessException
  309.             }
  310.         }
  311.         return obj;
  312.     }

  313. }