org.mentabean.util.FindMethod Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of menta-bean Show documentation
Show all versions of menta-bean Show documentation
An query helper and simple CRUD ORM.
/*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
* MentaBean => http://www.mentabean.org
* Author: Sergio Oliveira Jr. ([email protected])
*/
package org.mentabean.util;
import java.lang.reflect.Method;
import java.util.LinkedList;
import java.util.List;
/**
* Find method with polymorphism! Class.getMethod only finds an exact match.
*
* @author Jon Skeet (http://groups.google.com/group/comp.lang.java.programmer/browse_thread/thread/921ab91865c8cc2e/9e141d3d62e7cb3f)
*/
public class FindMethod {
/**
* Finds the most specific applicable method
*
* @param source
* Class to find method in
* @param name
* Name of method to find
* @param parameterTypes
* Parameter types to search for
*/
public static Method getMethod(Class extends Object> source, String name, Class extends Object>[] parameterTypes) throws NoSuchMethodException {
return internalFind(source.getMethods(), name, parameterTypes);
}
/**
* Finds the most specific applicable declared method
*
* @param source
* Class to find method in
* @param name
* Name of method to find
* @param parameterTypes
* Parameter types to search for
*/
public static Method getDeclaredMethod(Class extends Object> source, String name, Class extends Object>[] parameterTypes) throws NoSuchMethodException {
return internalFind(source.getDeclaredMethods(), name, parameterTypes);
}
/**
* Internal method to find the most specific applicable method
*/
private static Method internalFind(Method[] toTest, String name, Class extends Object>[] parameterTypes) throws NoSuchMethodException {
int l = parameterTypes.length;
// First find the applicable methods
List applicableMethods = new LinkedList();
for (int i = 0; i < toTest.length; i++) {
// Check the name matches
if (!toTest[i].getName().equals(name)) {
continue;
}
// Check the parameters match
Class extends Object>[] params = toTest[i].getParameterTypes();
if (params.length != l) {
continue;
}
int j;
for (j = 0; j < l; j++) {
if (!params[j].isAssignableFrom(parameterTypes[j])) {
break;
}
}
// If so, add it to the list
if (j == l) {
applicableMethods.add(toTest[i]);
}
}
/*
* If we've got one or zero methods, we can finish the job now.
*/
int size = applicableMethods.size();
if (size == 0) {
throw new NoSuchMethodException("No such method: " + name);
}
if (size == 1) {
return applicableMethods.get(0);
}
/*
* Now find the most specific method. Do this in a very primitive way - check whether each method is maximally specific. If more than one method is maximally specific, we'll throw an exception. For a definition of maximally specific, see JLS section 15.11.2.2.
*
* I'm sure there are much quicker ways - and I could probably set the second loop to be from i+1 to size. I'd rather not though, until I'm sure...
*/
int maximallySpecific = -1; // Index of maximally specific method
for (int i = 0; i < size; i++) {
int j;
// In terms of the JLS, current is T
Method current = applicableMethods.get(i);
Class extends Object>[] currentParams = current.getParameterTypes();
Class extends Object> currentDeclarer = current.getDeclaringClass();
for (j = 0; j < size; j++) {
if (i == j) {
continue;
}
// In terms of the JLS, test is U
Method test = applicableMethods.get(j);
Class extends Object>[] testParams = test.getParameterTypes();
Class extends Object> testDeclarer = test.getDeclaringClass();
// Check if T is a subclass of U, breaking if not
if (!testDeclarer.isAssignableFrom(currentDeclarer)) {
break;
}
// Check if each parameter in T is a subclass of the
// equivalent parameter in U
int k;
for (k = 0; k < l; k++) {
if (!testParams[k].isAssignableFrom(currentParams[k])) {
break;
}
}
if (k != l) {
break;
}
}
// Maximally specific!
if (j == size) {
if (maximallySpecific != -1) {
throw new NoSuchMethodException("Ambiguous method search - more " + "than one maximally specific method");
}
maximallySpecific = i;
}
}
if (maximallySpecific == -1) {
throw new NoSuchMethodException("No maximally specific method.");
}
return applicableMethods.get(maximallySpecific);
}
}