出现在mybtis中Reflector类中分类获取ger/Set方法时有如下代码信息:对于isBridge产生疑惑随记录
分析其主要作用如下:
/***
在Reflector.addUniqueMethods() 方法中会为每个方法生成唯 一签名,井记录到
uniqueMethods 集合中
***/
private void addUniqueMethods(Map uniqueMethods, Method[] methods) {
for (Method currentMethod : methods) {
if (!currentMethod.isBridge()) {
/***通过 Reflector.getSignature () 方法得到的方法签名是:返回值类型#方法名称:参
数类型列表。 例如,Reflector.getSignature(Method )方法的唯一签名是:
II java . lang.String#getSignature : ] ava.lang.reflect .Method
通过 Reflector. getSignature( )方法得到的方法签名是全局唯一的,可以作为该方法的唯一标识
***/
String signature = getSignature(currentMethod);
//检测是否在子类 中已经添加过该方法,如果在子类中已经添加过,9)1]表示子类覆盖了该方法,
//无需须再向 uniqueMethods 集合中添加该方法了
if (!uniqueMethods.containsKey(signature)) {
if (canAccessPrivateMethods()) {
try {
currentMethod.setAccessible(true);
} catch (Exception e) {
// Ignored. This is only a final precaution, nothing we can do.
}
}
//记录该签名和方 法的对应关系
uniqueMethods.put(signature, currentMethod);
}
}
}
}
具体代码信息请参看mybatis源码:Reflector类
在Reflector类中获取类中get/set方法一般会经过三个主要步骤:
1 )首先,调用 Reflector.getClassMethods()方法获取当前类以及其父类中定义的所有方法
的唯一签名 以及相应的 Method 对象 。而在获取时会调用Reflector.addUniqueMethods() 方法中会为每个方法生成唯 一签名,井记录到uniqueMethods 集合中( 2 )然后 ,按照 JavaBean 的规范,从 Reflector.getClassMethods()方法返回的 Method 数组中查找该类 中定义的 ge伽r 方法(具体哪些方法算是 ge忧er 方法,后面会详细介绍),将其记录到 conflictingGetters 集合中。conflictingGetters 集合( Has灿!fap
>()类型)的 key 为属性名称,value 是该属性对应的 getter 方法集合。 (3 调用 Reflector.resolveGetterConflicts()方法对这种覆写 的情况进行处理,同时会将处理得到的 getter 方法记录到 getMethods 集合,并将其返回值类型填充到 getTypes 集合 .
关于Method.isBridge原文地址:https://www.cnblogs.com/zsg88/p/7588929.html
桥接方法是 JDK 1.5 引入泛型后,为了使Java的泛型方法生成的字节码和 1.5 版本前的字节码相兼容而实现的。具体作用在于判断方法是否是有编译成在编译阶段自动生成的。
假定接口
public interface SuperClass {
void method(T t);
}
它的一个实现类
public class AClass implements SuperClass {
@Override
public void method(String s) {
System.out.println(s);
}
}
因为泛型是在1.5引入的,为了向前兼容,所以会在编译时去掉泛型(泛型擦除)。那么SuperClass接口中的method方法的参数在虚拟机中只能是Object。
它应该是这个样子:
public interface SuperClass {
void method(Object t);
}
而 AClass 实现了SuperClass 接口,但是它的实现方法却是:
public void method(String s) {
System.out.println(s);
}
根本就没有实现 void method(Object t) 方法。 这怎么回事,其实虚拟机自动实现了一个方法。
AClass在虚拟机中是这个样子:
public class AClass implements SuperClass { public void method(String s) { System.out.println(s); } public void method(Object s) { this.method((String) s); } }
这个void method(Object s) 就是桥接方法。
我们用这个命令查看
javap -p AClass.class
显示如下:
Compiled from "AClass.java"
public class AClass implements SuperClass {
public AClass();
public void method(java.lang.String);
public void method(java.lang.Object);
}
我们用反射写个测试,看结果如何
public static void main(String[] args) throws Exception {
AClass obj = new AClass();
Method m = AClass.class.getMethod("method", String.class);
m.invoke(obj, "XXXXXXXXXXXXXXXXXX");
System.out.println(m.isBridge());
m = AClass.class.getMethod("method", Object.class);
m.invoke(obj, "##################");
System.out.println(m.isBridge());
}
测试结果如下
XXXXXXXXXXXXXXXXXX
false
##################
true