io.microsphere.reflect.MethodUtils Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.microsphere.reflect;
import io.microsphere.logging.Logger;
import io.microsphere.util.ArrayUtils;
import io.microsphere.util.BaseUtils;
import javax.annotation.Nullable;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.TypeElement;
import javax.lang.model.util.Elements;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import java.util.StringJoiner;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.function.Predicate;
import static io.microsphere.collection.ListUtils.ofList;
import static io.microsphere.constants.SymbolConstants.COMMA_CHAR;
import static io.microsphere.constants.SymbolConstants.LEFT_PARENTHESIS_CHAR;
import static io.microsphere.constants.SymbolConstants.RIGHT_PARENTHESIS_CHAR;
import static io.microsphere.constants.SymbolConstants.SHARP_CHAR;
import static io.microsphere.lang.function.Predicates.and;
import static io.microsphere.lang.function.Streams.filterAll;
import static io.microsphere.logging.LoggerFactory.getLogger;
import static io.microsphere.reflect.AccessibleObjectUtils.trySetAccessible;
import static io.microsphere.reflect.MemberUtils.isPrivate;
import static io.microsphere.reflect.MemberUtils.isStatic;
import static io.microsphere.reflect.MethodUtils.MethodKey.buildKey;
import static io.microsphere.text.FormatUtils.format;
import static io.microsphere.util.AnnotationUtils.CALLER_SENSITIVE_ANNOTATION_CLASS;
import static io.microsphere.util.AnnotationUtils.isAnnotationPresent;
import static io.microsphere.util.ArrayUtils.EMPTY_CLASS_ARRAY;
import static io.microsphere.util.ClassUtils.getAllInheritedTypes;
import static io.microsphere.util.ClassUtils.getTypeName;
import static io.microsphere.util.ClassUtils.getTypes;
import static io.microsphere.util.ClassUtils.isArray;
import static io.microsphere.util.ClassUtils.isPrimitive;
import static java.util.Arrays.asList;
import static java.util.Collections.emptyList;
import static java.util.Collections.unmodifiableList;
/**
* The Java Reflection {@link Method} Utility class
*
* @author Mercy
* @since 1.0.0
*/
public abstract class MethodUtils extends BaseUtils {
private static final Logger logger = getLogger(MethodUtils.class);
/**
* The public methods of {@link Object}
*/
public final static List OBJECT_PUBLIC_METHODS = ofList(Object.class.getMethods());
/**
* The declared methods of {@link Object}
*/
public final static List OBJECT_DECLARED_METHODS = ofList(Object.class.getDeclaredMethods());
/**
* The {@link Predicate} reference to {@link MethodUtils#isObjectMethod(Method)}
*/
public final static Predicate super Method> OBJECT_METHOD_PREDICATE = MethodUtils::isObjectMethod;
/**
* The {@link Predicate} reference to {@link MemberUtils#isPublic(Member)}
*/
public final static Predicate super Method> PULIC_METHOD_PREDICATE = MemberUtils::isPublic;
private final static ConcurrentMap methodsCache = new ConcurrentHashMap<>(256);
private static final ConcurrentMap, Method[]> declaredMethodsCache = new ConcurrentHashMap<>(256);
/**
* Create an instance of {@link Predicate} for {@link Method} to exclude the specified declared class
*
* @param declaredClass the target class to exclude
* @return non-null
*/
public static Predicate super Method> excludedDeclaredClass(Class> declaredClass) {
return method -> !Objects.equals(declaredClass, method.getDeclaringClass());
}
/**
* Filter all {@link Method methods} of the target class by the specified {@link Predicate}
*
* @param targetClass the target class
* @param includeInheritedTypes include the inherited types, e,g. super classes or interfaces
* @param publicOnly only public method
* @param methodsToFilter (optional) the methods to be filtered
* @return non-null read-only {@link List}
*/
public static List filterMethods(Class> targetClass, boolean includeInheritedTypes, boolean publicOnly, Predicate super Method>... methodsToFilter) {
if (targetClass == null || isPrimitive(targetClass)) {
return emptyList();
}
if (isArray(targetClass)) {
return doFilterMethods(OBJECT_PUBLIC_METHODS, methodsToFilter);
}
if (Object.class.equals(targetClass)) {
return publicOnly ? doFilterMethods(OBJECT_PUBLIC_METHODS, methodsToFilter) : doFilterMethods(OBJECT_DECLARED_METHODS, methodsToFilter);
}
Predicate super Method> predicate = and(methodsToFilter);
if (publicOnly) {
predicate = predicate.and((Predicate) PULIC_METHOD_PREDICATE);
}
// All methods
List allMethods = new LinkedList<>();
if (includeInheritedTypes) {
while (targetClass != null) {
filterDeclaredMethodsHierarchically(targetClass, predicate, allMethods);
targetClass = targetClass.getSuperclass();
}
} else {
filterDeclaredMethods(targetClass, predicate, allMethods);
}
return unmodifiableList(allMethods);
}
/**
* Get all declared {@link Method methods} of the target class, excluding the inherited methods
*
* @param targetClass the target class
* @param methodsToFilter (optional) the methods to be filtered
* @return non-null read-only {@link List}
* @see #filterMethods(Class, boolean, boolean, Predicate[])
*/
public static List getDeclaredMethods(Class> targetClass, Predicate super Method>... methodsToFilter) {
return filterMethods(targetClass, false, false, methodsToFilter);
}
/**
* Get all public {@link Method methods} of the target class, including the inherited methods.
*
* @param targetClass the target class
* @param methodsToFilter (optional) the methods to be filtered
* @return non-null read-only {@link List}
* @see #filterMethods(Class, boolean, boolean, Predicate[])
*/
public static List getMethods(Class> targetClass, Predicate super Method>... methodsToFilter) {
return filterMethods(targetClass, false, true, methodsToFilter);
}
/**
* Get all declared {@link Method methods} of the target class, including the inherited methods.
*
* @param targetClass the target class
* @param methodsToFilter (optional) the methods to be filtered
* @return non-null read-only {@link List}
* @see #filterMethods(Class, boolean, boolean, Predicate[])
*/
public static List getAllDeclaredMethods(Class> targetClass, Predicate super Method>... methodsToFilter) {
return filterMethods(targetClass, true, false, methodsToFilter);
}
/**
* Get all public {@link Method methods} of the target class, including the inherited methods.
*
* @param targetClass the target class
* @param methodsToFilter (optional) the methods to be filtered
* @return non-null read-only {@link List}
* @see #filterMethods(Class, boolean, boolean, Predicate[])
*/
public static List getAllMethods(Class> targetClass, Predicate super Method>... methodsToFilter) {
return filterMethods(targetClass, true, true, methodsToFilter);
}
/**
* Find the {@link Method} by the specified type(including inherited types) and method name without the
* parameter type.
*
* @param targetClass the target type
* @param methodName the specified method name
* @return if not found, return null
*/
public static Method findMethod(Class targetClass, String methodName) {
return findMethod(targetClass, methodName, EMPTY_CLASS_ARRAY);
}
/**
* Find the {@link Method} by the specified type (including inherited types) and method name and parameter types
* with cache
*
* @param targetClass the target type
* @param methodName the method name
* @param parameterTypes the parameter types
* @return if not found, return null
*/
public static Method findMethod(Class targetClass, String methodName, Class>... parameterTypes) {
MethodKey key = buildKey(targetClass, methodName, parameterTypes);
return methodsCache.computeIfAbsent(key, MethodUtils::doFindMethod);
}
/**
* Find the declared {@link Method} by the specified type (including inherited types) and method name and parameter types
*
* @param targetClass the target type
* @param methodName the method name
* @param parameterTypes the parameter types
* @return
*/
public static Method findDeclaredMethod(Class> targetClass, String methodName, Class>... parameterTypes) {
if (targetClass == null) {
return null;
}
// First, try to find the declared method in directly target class
Method method = doFindDeclaredMethod(targetClass, methodName, parameterTypes);
if (method == null) { // Second, to find the declared method in the super class
Class> superClass = targetClass.getSuperclass();
method = findDeclaredMethod(superClass, methodName, parameterTypes);
}
if (method == null) { // Third, to find the declared method in the interfaces
for (Class> interfaceClass : targetClass.getInterfaces()) {
method = findDeclaredMethod(interfaceClass, methodName, parameterTypes);
if (method != null) {
break;
}
}
}
if (method == null) {
if (logger.isDebugEnabled()) {
logger.debug("The declared method was not found in the target class[name : '{}'] by name['{}'] and parameter types['{}']",
targetClass, methodName, Arrays.toString(parameterTypes));
}
}
return method;
}
/**
* Invoke the target objects' method
*
* @param object the target object
* @param methodName the method name
* @param arguments the method arguments
* @param the return type
* @return the target method's execution result
*/
public static R invokeMethod(Object object, String methodName, Object... arguments) {
Class type = object.getClass();
return invokeMethod(object, type, methodName, arguments);
}
/**
* Invoke the target classes' static method
*
* @param targetClass the target class
* @param methodName the method name
* @param arguments the method arguments
* @param the return type
* @return the target method's execution result
*/
public static R invokeStaticMethod(Class> targetClass, String methodName, Object... arguments) {
return invokeMethod(null, targetClass, methodName, arguments);
}
/**
* Invoke the target classes' static method
*
* @param method the method
* @param arguments the method arguments
* @param the return type
* @return the target method's execution result
*/
public static R invokeStaticMethod(Method method, Object... arguments) {
return invokeMethod(null, method, arguments);
}
public static R invokeMethod(Object instance, Class> type, String methodName, Object... arguments) {
Class[] parameterTypes = getTypes(arguments);
Method method = findMethod(type, methodName, parameterTypes);
if (method == null) {
throw new IllegalStateException(format("cannot find method[name : '{}'], class: '{}'", methodName, type.getName()));
}
return invokeMethod(instance, method, arguments);
}
/**
* Invokes the underlying method represented by this {@code Method}
* object, on the specified object with the specified parameters.
* Individual parameters are automatically unwrapped to match
* primitive formal parameters, and both primitive and reference
* parameters are subject to method invocation conversions as
* necessary.
*
* If the underlying method is static, then the specified {@code instance}
* argument is ignored. It may be null.
*
*
If the number of formal parameters required by the underlying method is
* 0, the supplied {@code args} array may be of length 0 or null.
*
*
If the underlying method is an instance method, it is invoked
* using dynamic method lookup as documented in The Java Language
* Specification, section {@jls 15.12.4.4}; in particular,
* overriding based on the runtime type of the target object may occur.
*
*
If the underlying method is static, the class that declared
* the method is initialized if it has not already been initialized.
*
*
If the method completes normally, the value it returns is
* returned to the caller of invoke; if the value has a primitive
* type, it is first appropriately wrapped in an object. However,
* if the value has the type of an array of a primitive type, the
* elements of the array are not wrapped in objects; in
* other words, an array of primitive type is returned. If the
* underlying method return type is void, the invocation returns
* null.
*
* @param instance the object the underlying method is invoked from
* @param method the underlying method
* @param arguments the arguments used for the method call
* @param
* @return the result of dispatching the method represented by
* this object on {@code instance} with parameters
* {@code arguments}
* @throws IllegalStateException if this {@code Method} object
* is enforcing Java language access control and the underlying
* method is inaccessible.
* @throws IllegalArgumentException if the method is an
* instance method and the specified object argument
* is not an instance of the class or interface
* declaring the underlying method (or of a subclass
* or implementor thereof); if the number of actual
* and formal parameters differ; if an unwrapping
* conversion for primitive arguments fails; or if,
* after possible unwrapping, a parameter value
* cannot be converted to the corresponding formal
* parameter type by a method invocation conversion.
* @throws RuntimeException if the underlying method
* throws an exception.
*/
public static R invokeMethod(@Nullable Object instance, Method method, Object... arguments) {
R result = null;
boolean accessible = false;
RuntimeException failure = null;
try {
trySetAccessible(method);
result = (R) method.invoke(instance, arguments);
} catch (IllegalAccessException e) {
String errorMessage = format("The method[signature : '{}' , instance : {}] can't be accessed[accessible : {}]", getSignature(method), instance, accessible);
failure = new IllegalStateException(errorMessage, e);
} catch (IllegalArgumentException e) {
String errorMessage = format("The arguments can't match the method[signature : '{}' , instance : {}] : {}", getSignature(method), instance, asList(arguments));
failure = new IllegalArgumentException(errorMessage, e);
} catch (InvocationTargetException e) {
String errorMessage = format("It's failed to invoke the method[signature : '{}' , instance : {} , arguments : {}]", getSignature(method), instance, asList(arguments));
failure = new RuntimeException(errorMessage, e.getTargetException());
}
if (failure != null) {
logger.error(failure.getMessage(), failure.getCause());
throw failure;
}
return result;
}
/**
* Tests whether one method, as a member of a given type,
* overrides another method.
*
* @param overrider the first method, possible overrider
* @param overridden the second method, possibly being overridden
* @return {@code true} if and only if the first method overrides
* the second
* @jls 8.4.8 Inheritance, Overriding, and Hiding
* @jls 9.4.1 Inheritance and Overriding
* @see Elements#overrides(ExecutableElement, ExecutableElement, TypeElement)
*/
public static boolean overrides(Method overrider, Method overridden) {
if (overrider == null || overridden == null) {
return false;
}
// equality comparison: If two methods are same
if (Objects.equals(overrider, overridden)) {
return false;
}
// Modifiers comparison: Any method must be non-static method
if (isStatic(overrider) || isStatic(overridden)) { //
return false;
}
// Modifiers comparison: the accessibility of any method must not be private
if (isPrivate(overrider) || isPrivate(overridden)) {
return false;
}
// Inheritance comparison: the target class of overrider must be inherit from the overridden's
if (!overridden.getDeclaringClass().isAssignableFrom(overrider.getDeclaringClass())) {
return false;
}
// Method comparison: must not be "default" method
if (overrider.isDefault()) {
return false;
}
// Method comparison: The method name must be equal
if (!Objects.equals(overrider.getName(), overridden.getName())) {
return false;
}
// Method comparison: The count of method parameters must be equal
if (!Objects.equals(overrider.getParameterCount(), overridden.getParameterCount())) {
return false;
}
// Method comparison: Any parameter type of overrider must equal the overridden's
for (int i = 0; i < overrider.getParameterCount(); i++) {
if (!Objects.equals(overridden.getParameterTypes()[i], overrider.getParameterTypes()[i])) {
return false;
}
}
// Method comparison: The return type of overrider must be inherit from the overridden's
return overridden.getReturnType().isAssignableFrom(overrider.getReturnType());
// Throwable comparison: "throws" Throwable list will be ignored, trust the compiler verify
}
/**
* Find the nearest overridden {@link Method method} from the inherited class
*
* @param overrider the overrider {@link Method method}
* @return if found, the overrider method
, or null
*/
public static Method findNearestOverriddenMethod(Method overrider) {
Class> targetClass = overrider.getDeclaringClass();
Method overriddenMethod = null;
for (Class> inheritedType : getAllInheritedTypes(targetClass)) {
overriddenMethod = findOverriddenMethod(overrider, inheritedType);
if (overriddenMethod != null) {
break;
}
}
return overriddenMethod;
}
/**
* Find the overridden {@link Method method} from the target class
*
* @param overrider the overrider {@link Method method}
* @param targetClass the class that is declaring the overridden {@link Method method}
* @return if found, the overrider method
, or null
*/
public static Method findOverriddenMethod(Method overrider, Class> targetClass) {
List matchedMethods = getAllMethods(targetClass, method -> overrides(overrider, method));
return matchedMethods.isEmpty() ? null : matchedMethods.get(0);
}
/**
* Get the signature of {@link Method the specified method}
*
* @param method {@link Method the specified method}
* @return non-null
*/
public static String getSignature(Method method) {
Class> targetClass = method.getDeclaringClass();
Class>[] parameterTypes = method.getParameterTypes();
int parameterCount = parameterTypes.length;
String[] parameterTypeNames = new String[parameterCount];
String methodName = method.getName();
String declaringClassName = getTypeName(targetClass);
int size = declaringClassName.length() + 1 // '#'
+ methodName.length() + 1 // '('
+ (parameterCount == 0 ? 0 : parameterCount - 1) // (parameterCount - 1) * ','
+ 1 // ')'
;
for (int i = 0; i < parameterCount; i++) {
Class> parameterType = parameterTypes[i];
String parameterTypeName = getTypeName(parameterType);
parameterTypeNames[i] = parameterTypeName;
size += parameterTypeName.length();
}
StringBuilder signatureBuilder = new StringBuilder(size);
signatureBuilder.append(declaringClassName).append(SHARP_CHAR).append(methodName).append(LEFT_PARENTHESIS_CHAR);
for (int i = 0; i < parameterCount; i++) {
String parameterTypeName = parameterTypeNames[i];
signatureBuilder.append(parameterTypeName);
if (i < parameterCount - 1) {
signatureBuilder.append(COMMA_CHAR);
}
parameterTypeNames[i] = null;
}
signatureBuilder.append(RIGHT_PARENTHESIS_CHAR);
return signatureBuilder.toString();
}
public static boolean isObjectMethod(Method method) {
if (method != null) {
return Objects.equals(Object.class, method.getDeclaringClass());
}
return false;
}
/**
* Test whether the specified {@link Method method} annotates {@linkplain jdk.internal.reflect.CallerSensitive} or not
*
* @param method {@link Method}
* @return true
if the specified {@link Method method} annotates {@linkplain jdk.internal.reflect.CallerSensitive}
* @see jdk.internal.reflect.CallerSensitive
*/
public static boolean isCallerSensitiveMethod(Method method) {
return isAnnotationPresent(method, CALLER_SENSITIVE_ANNOTATION_CLASS);
}
static void filterDeclaredMethodsHierarchically(Class> targetClass, Predicate super Method> methodToFilter, List methodsToCollect) {
filterDeclaredMethods(targetClass, methodToFilter, methodsToCollect);
for (Class> interfaceClass : targetClass.getInterfaces()) {
filterDeclaredMethodsHierarchically(interfaceClass, methodToFilter, methodsToCollect);
}
}
static void filterDeclaredMethods(Class> targetClass, Predicate super Method> methodToFilter, List methodsToCollect) {
for (Method method : doGetDeclaredMethods(targetClass)) {
if (methodToFilter.test(method)) {
methodsToCollect.add(method);
}
}
}
static Method doFindDeclaredMethod(Class> klass, String methodName, Class>[] parameterTypes) {
Method[] declaredMethods = doGetDeclaredMethods(klass);
return doFindMethod(declaredMethods, methodName, parameterTypes);
}
static Method doFindMethod(Method[] methods, String methodName, Class>[] parameterTypes) {
Method targetMethod = null;
for (Method method : methods) {
if (matches(method, methodName, parameterTypes)) {
targetMethod = method;
break;
}
}
if (logger.isTraceEnabled()) {
logger.trace("The target method[name : '{}' , parameter types : '{}'] can't be found in the methods : {}",
methodName, Arrays.toString(parameterTypes), Arrays.toString(methods));
}
return targetMethod;
}
static boolean matches(Method method, String methodName, Class>[] parameterTypes) {
return Objects.equals(method.getName(), methodName)
&& Arrays.equals(method.getParameterTypes(), parameterTypes);
}
static Method[] doGetDeclaredMethods(Class> klass) {
return declaredMethodsCache.computeIfAbsent(klass, c -> c.getDeclaredMethods());
}
static List doFilterMethods(List methods, Predicate super Method>... methodsToFilter) {
return unmodifiableList(filterAll(methods, methodsToFilter));
}
static Method doFindMethod(MethodKey key) {
Class> declaredClass = key.declaredClass;
String methodName = key.methodName;
Class>[] parameterTypes = key.parameterTypes;
return findDeclaredMethod(declaredClass, methodName, parameterTypes);
}
static class MethodKey {
private final Class> declaredClass;
private final String methodName;
private final Class>[] parameterTypes;
MethodKey(Class> declaredClass, String methodName, Class>[] parameterTypes) {
this.declaredClass = declaredClass;
this.methodName = methodName;
this.parameterTypes = parameterTypes == null ? EMPTY_CLASS_ARRAY : parameterTypes;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
MethodKey methodKey = (MethodKey) o;
if (!Objects.equals(declaredClass, methodKey.declaredClass)) return false;
if (!Objects.equals(methodName, methodKey.methodName)) return false;
// Probably incorrect - comparing Object[] arrays with Arrays.equals
return Arrays.equals(parameterTypes, methodKey.parameterTypes);
}
@Override
public int hashCode() {
int result = declaredClass != null ? declaredClass.hashCode() : 0;
result = 31 * result + (methodName != null ? methodName.hashCode() : 0);
result = 31 * result + Arrays.hashCode(parameterTypes);
return result;
}
@Override
public String toString() {
StringJoiner stringJoiner = new StringJoiner(",", "(", ") ");
ArrayUtils.forEach(parameterTypes, parameterType -> stringJoiner.add(getTypeName(parameterType)));
return getTypeName(declaredClass) + "#" + methodName + stringJoiner;
}
static MethodKey buildKey(Class> declaredClass, String methodName, Class>[] parameterTypes) {
return new MethodKey(declaredClass, methodName, parameterTypes);
}
}
}