Mybatis 源码学习获取接口代理对象

本文深入剖析MyBatis框架中的动态代理机制,包括MapperRegistry、MapperProxyFactory及MapperProxy等核心组件的工作原理,展示了如何利用Java反射机制实现Mapper接口的代理对象。
摘要由CSDN通过智能技术生成

<T> T getMapper(Class<T> type) 实现;使用 MapperInteface 泛指我们自定义的 mapper 接口;

主要组件类:

org.apache.ibatis.session.SqlSession

org.apache.ibatis.session.defaults.DefaultSqlSession

org.apache.ibatis.session.Configuration

org.apache.ibatis.binding.MapperRegistry

org.apache.ibatis.binding.MapperProxyFactory<T>

org.apache.ibatis.binding.MapperProxy<T>

org.apache.ibatis.binding.MapperMethod

 

MapperRegistry 中保存了 MapperInteface.class 和  MapperProxyFactory<MapperInteface> 的对应关系 :


Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<Class<?>, MapperProxyFactory<?>>();

 

MapperProxyFactory 创建 MapperInteface 的代理对象  ,  使用  jdk java.lang.reflect.Proxy 实现 ;

protected T newInstance(MapperProxy<T> mapperProxy) {
    return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);
}

public T newInstance(SqlSession sqlSession) {
  final MapperProxy<T> mapperProxy = new MapperProxy<T>(sqlSession, mapperInterface, methodCache);
  return newInstance(mapperProxy);
}

 

MapperProxy 实现了 java.lang.reflect.InvocationHandler 接口 , 代理对象在使用时会去调用 MapperProxy  的 invoke 方法 ;

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
      if (Object.class.equals(method.getDeclaringClass())) {
        return method.invoke(this, args);
      } else if (isDefaultMethod(method)) {
        return invokeDefaultMethod(proxy, method, args);
      }
    } catch (Throwable t) {
      throw ExceptionUtil.unwrapThrowable(t);
    }
    final MapperMethod mapperMethod = cachedMapperMethod(method);
    return mapperMethod.execute(sqlSession, args);
}

 

整个过程并没有被代理对象 , 一般情况下我们会把被代理对象组合进 InvocationHandler  的实现类中 , 在mybatis 中也就是 MapperProxy  中 , 但是 MapperProxy  中并没有这样一个被代理对象 , 因为整个设计中就没有它的存在 。 mybatis 中执行后续流程的是 MapperMethod , MapperMethod  实例对应 MapperInteface 中的每一个方法 , MapperMethod 的实例会被缓存在 MapperProxy  中。

实现整个过程的核心技术就是 动态代理,mybatis 使用了 jdk 的动态代理 , 对于处理 MapperInterface 这样的接口来说在合适不过了 , 还有一个问题 , Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)  方法的返回值是 Object 类型 , 但是我们发现我们在使用 mybatis 时并没有做强制类型转换 , 因为 MapperProxyFactory<T> , MapperProxy<T> 都使用了泛型 , 并且他们的泛型要保持一致。

 

转载于:https://my.oschina.net/j4love/blog/1570489

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值