本文將分析mybatis與spring整合的MapperScannerConfigurer的底層原理,以前已經分析過Java中實現動態,可使用jdk自帶api和cglib第三方庫生成動態代理。本文分析的mybatis版本3.2.7,mybatis-spring版本1.2.2。 html
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
以如下代碼爲示例進行講解(部分代碼,其餘代碼及配置省略): 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實現了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動態代碼淺析和本文。