All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.expression.spel.support.ReflectiveMethodResolver Maven / Gradle / Ivy

/*
 * Copyright 2002-2010 the original author or authors.
 *
 * Licensed 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 org.springframework.expression.spel.support;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.springframework.core.MethodParameter;
import org.springframework.core.convert.TypeDescriptor;
import org.springframework.expression.AccessException;
import org.springframework.expression.EvaluationContext;
import org.springframework.expression.EvaluationException;
import org.springframework.expression.MethodExecutor;
import org.springframework.expression.MethodFilter;
import org.springframework.expression.MethodResolver;
import org.springframework.expression.TypeConverter;
import org.springframework.expression.spel.SpelEvaluationException;
import org.springframework.expression.spel.SpelMessage;
import org.springframework.util.CollectionUtils;

/**
 * A method resolver that uses reflection to locate the method that should be invoked.
 *
 * @author Andy Clement
 * @author Juergen Hoeller
 * @since 3.0
 */
public class ReflectiveMethodResolver implements MethodResolver {

	private static Method[] NO_METHODS = new Method[0];

	private Map, MethodFilter> filters = null;

	
	/**
	 * Locate a method on a type. There are three kinds of match that might occur:
	 * 
    *
  1. An exact match where the types of the arguments match the types of the constructor *
  2. An in-exact match where the types we are looking for are subtypes of those defined on the constructor *
  3. A match where we are able to convert the arguments into those expected by the constructor, * according to the registered type converter. *
*/ public MethodExecutor resolve(EvaluationContext context, Object targetObject, String name, List argumentTypes) throws AccessException { try { TypeConverter typeConverter = context.getTypeConverter(); Class type = (targetObject instanceof Class ? (Class) targetObject : targetObject.getClass()); Method[] methods = type.getMethods(); // If a filter is registered for this type, call it MethodFilter filter = (this.filters != null ? this.filters.get(type) : null); if (filter != null) { List methodsForFiltering = new ArrayList(); for (Method method: methods) { methodsForFiltering.add(method); } List methodsFiltered = filter.filter(methodsForFiltering); if (CollectionUtils.isEmpty(methodsFiltered)) { methods = NO_METHODS; } else { methods = methodsFiltered.toArray(new Method[methodsFiltered.size()]); } } Arrays.sort(methods, new Comparator() { public int compare(Method m1, Method m2) { int m1pl = m1.getParameterTypes().length; int m2pl = m2.getParameterTypes().length; return (new Integer(m1pl)).compareTo(m2pl); } }); Method closeMatch = null; int[] argsToConvert = null; Method matchRequiringConversion = null; boolean multipleOptions = false; for (Method method : methods) { if (method.isBridge()) { continue; } if (method.getName().equals(name)) { Class[] paramTypes = method.getParameterTypes(); List paramDescriptors = new ArrayList(paramTypes.length); for (int i = 0; i < paramTypes.length; i++) { paramDescriptors.add(new TypeDescriptor(new MethodParameter(method, i))); } ReflectionHelper.ArgumentsMatchInfo matchInfo = null; if (method.isVarArgs() && argumentTypes.size() >= (paramTypes.length - 1)) { // *sigh* complicated matchInfo = ReflectionHelper.compareArgumentsVarargs(paramDescriptors, argumentTypes, typeConverter); } else if (paramTypes.length == argumentTypes.size()) { // name and parameter number match, check the arguments matchInfo = ReflectionHelper.compareArguments(paramDescriptors, argumentTypes, typeConverter); } if (matchInfo != null) { if (matchInfo.kind == ReflectionHelper.ArgsMatchKind.EXACT) { return new ReflectiveMethodExecutor(method, null); } else if (matchInfo.kind == ReflectionHelper.ArgsMatchKind.CLOSE) { closeMatch = method; } else if (matchInfo.kind == ReflectionHelper.ArgsMatchKind.REQUIRES_CONVERSION) { if (matchRequiringConversion != null) { multipleOptions = true; } argsToConvert = matchInfo.argsRequiringConversion; matchRequiringConversion = method; } } } } if (closeMatch != null) { return new ReflectiveMethodExecutor(closeMatch, null); } else if (matchRequiringConversion != null) { if (multipleOptions) { throw new SpelEvaluationException(SpelMessage.MULTIPLE_POSSIBLE_METHODS, name); } return new ReflectiveMethodExecutor(matchRequiringConversion, argsToConvert); } else { return null; } } catch (EvaluationException ex) { throw new AccessException("Failed to resolve method", ex); } } public void registerMethodFilter(Class type, MethodFilter filter) { if (this.filters == null) { this.filters = new HashMap, MethodFilter>(); } if (filter == null) { this.filters.remove(type); } else { this.filters.put(type,filter); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy