isBridge-----Java反射中桥接方法

 出现在mybtis中Reflector类中分类获取ger/Set方法时有如下代码信息:对于isBridge产生疑惑随记录

分析其主要作用如下:

     

/***
  在Reflector.addUniqueMethods() 方法中会为每个方法生成唯 一签名,井记录到
uniqueMethods 集合中
***/ 
private void addUniqueMethods(Map<String, Method> 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<Stri吨,List<Method>>()类型)的 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<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

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值