MyBatis 的 DAO 接口跟 XML 文件里面的 SQL 是如何建立关系的
MyBatis 会先解析这些 XML 文件,通过 XML 文件里面的命名空间 (namespace)跟 DAO 建立关系;然后 XML 中的每段 SQL 会有一个id 跟 DAO 中的接口进行关联。
首先我们要知道每个基于 MyBatis 的应用都是以一个 SqlSessionFactory
的实例为中心的,SqlSessionFactory
的实例可以通过 SqlSessionFactoryBuilder
获得。
但 SqlSessionFactory
是一个接口,它里面其实就两个方法:openSession
、getConfiguration
其中,openSession
方法是为了获取一个 SqlSession
对象,完成必要数据库增删改查功能。但是,SqlSessionFactory
属性太少了,所以需要 getConfiguration
的配合;来配置mapper
映射文件、SQL 参数、返回值类型、缓存等属性。
/** * Creates an {@link SqlSession} out of a connection or a DataSource * * @author Clinton Begin */ public interface SqlSessionFactory { SqlSession openSession(); SqlSession openSession(boolean autoCommit); SqlSession openSession(Connection connection); SqlSession openSession(TransactionIsolationLevel level); SqlSession openSession(ExecutorType execType); SqlSession openSession(ExecutorType execType, boolean autoCommit); SqlSession openSession(ExecutorType execType, TransactionIsolationLevel level); SqlSession openSession(ExecutorType execType, Connection connection); Configuration getConfiguration(); }
可以看到 getConfiguration
是属于 Configuration
类的一个方法。你可以把它当成一个配置管家。MyBatis 所有的配置信息都维持在 Configuration
对象之中,基本每个对象都会持有它的引用。
但日常开发中我们都是将 MyBatis 与 Spring 一起使用的,所以把实例化交给 Spring 处理。
因此我们可以看下 org.MyBatis.spring.SqlSessionFactoryBean
,它实现了 InitializingBean
接口。这说明,在这个类被实例化之后会调用到 afterPropertiesSet()
。它只有一个方法
public void afterPropertiesSet() throws Exception { this.sqlSessionFactory = buildSqlSessionFactory(); }
而这个 afterPropertiesSet
方法只有一个动作,就是 buildSqlSessionFactory
。它可以分为两部分来看:
- 1、从配置文件的
property
属性中加载各种组件,解析配置到configuration
中 - 2、加载 mapper 文件,解析 SQL 语句,封装成
MappedStatement
对象,配置到configuration
中。
mapper 接口方法是怎样被调用到的?
大致有如下两种方式:
- MyBatis 提供的 API
使用 MyBatis 提供的 API 进行操作,通过获取 SqlSession
对象,然后根据 Statement Id 和参数来操作数据库。
String statement = "com.mmzsblog.business.DAO.MemberMapper.getMemberList";
List<Member> result = sqlsession.selectList(statement);
- mapper 接口
定义 Mapper 接口,并在里面定义一系列业务数据操作方法。在 Service 层通过注入 mapper 属性,调用其方法就可以执行数据库操作。
public interface MemberMapper { List<Member> getMemberList(); } @Service public class MemberServiceImpl implements MemberService{ @Resource private MemberMapper memberMapper; @Override public List<Member> getMemberList() { return memberMapper.getMemberList(); } }
Mapper 接口的代理创建过程
首先我们会配置需要扫描的基本包路径
通过注解的方式配置@MapperScan({"com.mmzsblog.business.DAO"})
或者xml的方式配置:
<bean class="org.MyBatis.spring.mapper.MapperScannerConfigurer">
<property name="basePackage" value="com.mmzsblog.business.DAO" />
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"></property>
</bean>
开始扫描
来到 org.MyBatis.spring.mapper.MapperScannerConfigurer
这个类,可以看到它实现了几个接口
其中的重点是 BeanDefinitionRegistryPostProcessor
。它可以动态的注册 Bean 信息,方法为 postProcessBeanDefinitionRegistry()
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) { if (this.processPropertyPlaceHolders) { this.processPropertyPlaceHolders(); } // 创建ClassPath扫描器,设置属性,然后调用扫描方法 ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry); scanner.setAddToConfig(this.addToConfig); scanner.setAnnotationClass(this.annotationClass); scanner.setMarkerInterface(this.markerInterface); scanner.setSqlSessionFactory(this.sqlSessionFactory); scanner.setSqlSessionTemplate(this.sqlSessionTemplate); scanner.setSqlSessionFactoryBeanName(this.sqlSessionFactoryBeanName); scanner.setSqlSessionTemplateBeanName(this.sqlSessionTemplateBeanName); scanner.setResourceLoader(this.applicationContext); scanner.setBeanNameGenerator(this.nameGenerator); // 创建ClassPath扫描器,设置属性,然后调用扫描方法 scanner.registerFilters(); scanner.scan(StringUtils.tokenizeToStringArray(this.basePackage, ",; \t\n")); }
ClassPathMapperScanner
继承自 Spring 中的类 ClassPathBeanDefinitionScanner
,所以它的 scan 方法会调用到父类 ClassPathBeanDefinitionScanner
的 scan 方法
public class ClassPathBeanDefinitionScanner extends ClassPathScanningCandidateComponentProvider { …… public int scan(String... basePackages) { // int beanCountAtScanStart = this.registry.getBeanDefinitionCount(); this.doScan(basePackages); if (this.includeAnnotationConfig) { AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry); } return this.registry.getBeanDefinitionCount() - beanCountAtScanStart; } …… }
而在父类的 scan 方法中又调用到子类 ClassPathMapperScanner
重写的 doScan
方法。
public class ClassPathMapperScanner extends ClassPathBeanDefinitionScanner { …… public Set<BeanDefinitionHolder> doScan(String... basePackages) { Set<BeanDefinitionHolder> beanDefinitions = super.doScan(basePackages); if (beanDefinitions.isEmpty()) { this.logger.warn("No MyBatis mapper was found in '" + Arrays.toString(basePackages) + "' package. Please check your configuration."); } else { this.processBeanDefinitions(beanDefinitions); } return beanDefinitions; } …… }
此处 super.doScan(basePackages)
是 Spring 中的方法
bean 注册完成并创建 sqlSession 代理
并且经过上面这些步骤,此时已经扫描到了所有的 Mapper 接口,并将其注册为 BeanDefinition
对象。而注册的时候就是用到了上面 doScan
方法中的 processBeanDefinitions
方法
public class ClassPathMapperScanner extends ClassPathBeanDefinitionScanner { …… // 设置 beanClass private MapperFactoryBean<?> mapperFactoryBean = new MapperFactoryBean(); …… private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) { Iterator var3 = beanDefinitions.iterator(); while(var3.hasNext()) { BeanDefinitionHolder holder = (BeanDefinitionHolder)var3.next(); GenericBeanDefinition definition = (GenericBeanDefinition)holder.getBeanDefinition(); if (this.logger.isDebugEnabled()) { this.logger.debug("Creating MapperFactoryBean with name '" + holder.getBeanName() + "' and '" + definition.getBeanClassName() + "' mapperInterface"); } // 将 mapper 接口的名称添加到构造参数 definition.getConstructorArgumentValues().addGenericArgumentValue(definition.getBeanClassName()); // 设置 BeanDefinition的class definition.setBeanClass(this.mapperFactoryBean.getClass()); // 添加属性 addToConfig definition.getPropertyValues().add("addToConfig", this.addToConfig); boolean explicitFactoryUsed = false; // 添加属性 sqlSessionFactory if (StringUtils.hasText(this.sqlSessionFactoryBeanName)) { definition.getPropertyValues().add("sqlSessionFactory", new RuntimeBeanReference(this.sqlSessionFactoryBeanName)); explicitFactoryUsed = true; } else if (this.sqlSessionFactory != null) { definition.getPropertyValues().add("sqlSessionFactory", this.sqlSessionFactory); explicitFactoryUsed = true; } if (StringUtils.hasText(this.sqlSessionTemplateBeanName)) { if (explicitFactoryUsed) { this.logger.warn("Cannot use both: sqlSessionTemplate and sqlSessionFactory together. sqlSessionFactory is ignored."); } definition.getPropertyValues().add("sqlSessionTemplate", new RuntimeBeanReference(this.sqlSessionTemplateBeanName)); explicitFactoryUsed = true; } else if (this.sqlSessionTemplate != null) { if (explicitFactoryUsed) { this.logger.warn("Cannot use both: sqlSessionTemplate and sqlSessionFactory together. sqlSessionFactory is ignored."); } definition.getPropertyValues().add("sqlSessionTemplate", this.sqlSessionTemplate); explicitFactoryUsed = true; } if (!explicitFactoryUsed) { if (this.logger.isDebugEnabled()) { this.logger.debug("Enabling autowire by type for MapperFactoryBean with name '" + holder.getBeanName() + "'."); } definition.setAutowireMode(2); } } } …… }
处理的过程相对比较简单,只是往 BeanDefinition
对象中设置了一些属性。例如:
- 设置 beanClass
设置 BeanDefinition
对象的 BeanClass
为 MapperFactoryBean<?>
。这就相当于使用 MemberMapper
注册时:当前的 mapper 接口在 Spring 容器中,beanName 是 memberMapper
,beanClass 是 MapperFactoryBean.class
。故在Spring 的 IOC 初始化的时候,实例化的对象就是 MapperFactoryBean
对象。
- 设置
sqlSessionFactory
属性
为 BeanDefinition
对象添加属性 sqlSessionFactory
,是为了 BeanDefinition
对象设置PropertyValue
的时候,方便调用到 setSqlSessionFactory()
。
创建 sqlSession 代理类
最终在 setSqlSessionFactory
这个方法里,sqlSession
获取到的是 SqlSessionTemplate
实例。而在 SqlSessionTemplate
对象中,主要包含 sqlSessionFactory
和sqlSessionProxy
,而 sqlSessionProxy
实际上是 SqlSession
接口的代理对象。实际调用的是代理类的 invoke
方法
public class MapperProxy<T> implements InvocationHandler, Serializable { …… @Override 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); } …… }
总结
Mapper 接口的代理创建过程大致如下:
- 1、扫描 mapper 接口基本包路径下的所有对象,将其注册为
BeanDefinition
对象 - 2、设置
BeanDefinition
的对象的 beanClass 和sqlSessionFactory
属性(而其中获取BeanDefinition
对象的时候,调用其工厂方法getObject
,返回 mapper 接口的代理类) - 3、设置
sqlSessionFactory
属性的时候,会调用SqlSessionTemplate
的构造方法,创建SqlSession
接口的代理类
最后我们在 Service 层注入属性的时候,返回的就是代理类。执行 memberDao 的方法的时候,实际调用的也是代理类的 invoke 方法。
解释建立关系
MyBatis 在初始化 SqlSessionFactoryBean 的时候,找到配置需要扫描的基本包路径去解析里面所有的 XML 文件。重点就在如下两个地方
创建 SqlSource
MyBatis 会把每个 SQL 标签封装成 SqlSource 对象。然后根据 SQL 语句的不同,又分为动态 SQL 和静态 SQL 。其中,静态 SQL 包含一段 String 类型的 SQL 语句;而动态 SQL 则是由一个个 SqlNode 组成
创建 MappedStatement
XML 文件中的每一个 SQL 标签就对应一个 MappedStatement 对象,这里面有两个属性很重要。
- id
全限定类名+方法名组成的 ID。
- sqlSource
当前 SQL 标签对应的 SqlSource 对象。创建完 MappedStatement
对象,会将它缓存到 Configuration#mappedStatements
中。
前面初始化中提到的 Configuration
对象,我们知道它就是 MyBatis 中的配置大管家,基本所有的配置信息都维护在这里
<!-- namespace 的值就是全限定类名 --> <mapper namespace="com.java.mmzsblog.DAO.MemberMapper"> …… <!-- select 标签中 id 的值就是方法名,它和全限定类中的方法名是对应的 --> <select id="getMemberById" resultType="com.java.mmzsblog.entity.member"> select * from member <where> <if test="memberId!=null"> and member_id=#{memberId} </if> </where> </select> …… </mapper>
把所有的 XML 都解析完成之后,Configuration
就包含了所有的 SQL 信息。然后解析完成的 XML 大概就是这样了
当我们执行 MyBatis 方法的时候,就通过全限定类名+方法名
找到 MappedStatement
对象,然后解析里面的 SQL 内容,执行即可