桥接方法是 JDK 1.5 引入泛型后,为了使Java的泛型方法生成的字节码和 1.5 版本前的字节码相兼容,由编译器自动生成的方法。我们可以通过Method.isBridge()方法来判断一个方法是否是桥接方法。
假定接口
public interface SuperClass<T> {
void method(T t);
}
它的一个实现类
public class AClass implements SuperClass<String> {
@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<java.lang.String> {
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