Spring與Mybatis整合的MapperScannerConfigurer

本文將分析mybatis與spring整合的MapperScannerConfigurer的底層原理,以前已經分析過Java中實現動態,可使用jdk自帶api和cglib第三方庫生成動態代理。本文分析的mybatis版本3.2.7,mybatis-spring版本1.2.2。 html

MapperScannerConfigurer介紹

MapperScannerConfigurer是spring和mybatis整合的mybatis-spring jar包中提供的一個類。 java

想要了解該類的做用,就得先了解MapperFactoryBean。 git

MapperFactoryBean的出現爲了代替手工使用SqlSessionDaoSupport或SqlSessionTemplate編寫數據訪問對象(DAO)的代碼,使用動態代理實現。 github

好比下面這個官方文檔中的配置: spring

<bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
  <property name="mapperInterface" value=http://www.cnblogs.com/fangjian0423/p/"org.mybatis.spring.sample.mapper.UserMapper" />

org.mybatis.spring.sample.mapper.UserMapper是一個接口,咱們建立一個MapperFactoryBean實例,而後注入這個接口和sqlSessionFactory(mybatis中提供的SqlSessionFactory接口,MapperFactoryBean會使用SqlSessionFactory建立SqlSession)這兩個屬性。 sql

以後想使用這個UserMapper接口的話,直接經過spring注入這個bean,而後就能夠直接使用了,spring內部會建立一個這個接口的動態代理。 api

當發現要使用多個MapperFactoryBean的時候,一個一個定義確定很是麻煩,因而mybatis-spring提供了MapperScannerConfigurer這個類,它將會查找類路徑下的映射器並自動將它們建立成MapperFactoryBean。 mybatis

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
  <property name="basePackage" value=http://www.cnblogs.com/fangjian0423/p/"org.mybatis.spring.sample.mapper" />

這段配置會掃描org.mybatis.spring.sample.mapper下的全部接口,而後建立各自接口的動態代理類。 app

MapperScannerConfigurer底層代碼分析

以如下代碼爲示例進行講解(部分代碼,其餘代碼及配置省略): post

package org.format.dynamicproxy.mybatis.dao;
public interface UserDao {
    public User getById(int id);
    public int add(User user);    
    public int update(User user);    
    public int delete(User user);    
    public List<User> getAll();    
}

<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
    <property name="basePackage" value=http://www.cnblogs.com/fangjian0423/p/"org.format.dynamicproxy.mybatis.dao"/>

咱們先經過測試用例debug查看userDao的實現類究竟是什麼。

咱們能夠看到,userDao是1個MapperProxy類的實例。
看下MapperProxy的源碼,沒錯,實現了InvocationHandler,說明使用了jdk自帶的動態代理。

public class MapperProxy<T> implements InvocationHandler, Serializable {

  private static final long serialversionUID = -6424540398559729838L;
  private final SqlSession sqlSession;
  private final Class<T> mapperinterface;
  private final Map<Method, MapperMethod> methodCache;

  public MapperProxy(SqlSession sqlSession, Class<T> mapperInterface, Map<Method, MapperMethod> methodCache) {
    this.sqlSession = sqlSession;
    this.mapperInterface = mapperInterface;
    this.methodCache = methodCache;
  }

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

  private MapperMethod cachedMapperMethod(Method method) {
    MapperMethod mapperMethod = methodCache.get(method);
    if (mapperMethod == null) {
      mapperMethod = new MapperMethod(mapperInterface, method, sqlSession.getConfiguration());
      methodCache.put(method, mapperMethod);
    }
    return mapperMethod;
  }

}

下面開始分析MapperScannerConfigurer的源碼

MapperScannerConfigurer實現了BeanDefinitionRegistryPostProcessor接口,BeanDefinitionRegistryPostProcessor接口是一個能夠修改spring工長中已定義的bean的接口,該接口有個postProcessBeanDefinitionRegistry方法。

而後咱們看下ClassPathMapperScanner中的關鍵是如何掃描對應package下的接口的。

其實MapperScannerConfigurer的做用也就是將對應的接口的類型改造爲MapperFactoryBean,而這個MapperFactoryBean的屬性mapperInterface是原類型。MapperFactoryBean本文開頭已分析過。

因此最終咱們仍是要分析MapperFactoryBean的實現原理!

MapperFactoryBean繼承了SqlSessionDaoSupport類,SqlSessionDaoSupport類繼承DaoSupport抽象類,DaoSupport抽象類實現了InitializingBean接口,所以實例個MapperFactoryBean的時候,都會調用InitializingBean接口的afterPropertiesSet方法。

DaoSupport的afterPropertiesSet方法:

MapperFactoryBean重寫了checkDaoConfig方法:

而後經過spring工廠拿對應的bean的時候:

這裏的SqlSession是SqlSessionTemplate,SqlSessionTemplate的getMapper方法:

Configuration的getMapper方法,會使用MapperRegistry的getMapper方法:

MapperRegistry的getMapper方法:

MapperProxyFactory構造MapperProxy:

沒錯! MapperProxyFactory就是使用了jdk組帶的Proxy完成動態代理。
MapperProxy原本一開始已經提到。MapperProxy內部使用了MapperMethod類完成方法的調用:

下面,咱們以UserDao的getById方法來debug看看MapperMethod的execute方法是如何走的。

@Test
public void testGet() {
    int id = 1; system.out.println(userDao.getById(id));
}
<select id="getById" parameterType="int" resultType="org.format.dynamicproxy.mybatis.bean.User">
    SELECT * FROM users WHERE id = #{id}
</select>


示例代碼:https://github.com/fangjian0423/dynamic-proxy-mybatis-study

總結

來到了新公司,接觸了Mybatis,之前接觸過~ 可是接觸的不深刻,忽然發現spring與mybatis整合以後能夠只寫個接口而不實現,spring默認會幫咱們實現,而後以爲很是神奇,因而寫了篇java動態代碼淺析和本文。

相關文章
相關標籤/搜索