3
假设我们有以下的接口和实现:如何查找桥接方法的声明方法?
interface Matcher<T>{
boolean matches(T arg);
}
class NumberMatcher<T extends Number> implements Matcher<T>{
@Override
public boolean matches(T arg){...}
}
class StringMatcher extends Matcher<String>{
@Override
public boolean matches(String arg){ ...}
}
class CustomMatcher extends NumberMatcher<Integer> {
public boolean matches(String arg){...}
@Override
public boolean matches(Integer arg){...}
}
我需要的是一个给定的匹配器实现的匹配(T)方法的参数的类型。
NumberMatcher numberMatcher = new NumberMatcher<Long>();
StringMatcher stringMatcher = new StringMatcher();
CustomMatcher customMatcher = new CustomMatcher();
Matcher<Date> dateMatcher = new Matcher<Date>(){...};
getArgumentType(numberMatcher) // should return Number.class
getArgumentType(stringMatcher) // should return String.class
getArgumentType(customMatcher) // should return Integer.class
getArgumentType(dateMatcher) // should return Object.class
这里是除了CustomMatcher箱子一个可行的实施方式中,导致它未能检测到被覆盖的匹配(..)方法并返回String.class
代替Integer.class
。
Class<?> getArgumentType(Matcher<?> matcher) {
Method[] methods = matcher.getClass().getMethods();
for (Method method : methods) {
if (isMatchesMethod(method)) {
return method.getParameterTypes()[0];
}
}
throw new NoSuchMethodError("Method 'matches(T)' not found!");
}
private boolean isMatchesMethod(Method method) {
if (!isPublic(method.getModifiers()))
return false;
if (method.getParameterCount() != 1)
return false;
return method.getName().equals("matches");
}
编辑:
我要寻找的是不需要指定参数类型这样一个解决方案:只要你可以编辑
interface Matcher<T>{
boolean matches(T arg);
Class<T> argumentType();
}
仅供参考:显而易见的解决方案是将'getDeclaredAnnotations()'应用于该方法。但是这种情况在这种情况下不起作用,因为'@ Override'注释在运行时不再可用。 –
也许['Method.getDeclaringClass'](https://docs.oracle.com/javase/8/docs/api/java/lang/reflect/Method.html#getDeclaringClass--)会有帮助吗? – Seelenvirtuose
@Seelenvirtuose Method.getDeclaringClass()在'CustomMatcher'的情况下不起作用,因为声明类是'NumberMatcher'而不是'Matcher'接口 – Chriss