大家好,我是雨梦。
今天,跟大家分享一下,MyBatis的执行流程。在文章开始前,先给大家推荐一个属于我们程序员的干饭神器,“饥饿袋鼠”。用了一段时间,确实很方便,每顿饭饭都可以省好几块呢,下一顿饭饭可以尝试一下,下面的小卡片可以进入
前言
MyBatis可能很多人都一直在用,但是MyBatis的SQL执行流程可能并不是所有人都清楚了,那么既然进来了,通读本文你将收获如下:
1、Mapper接口和映射文件是如何进行绑定的2、MyBatis中SQL语句的执行流程3、自定义MyBatis中的参数设置处理器typeHandler4、自定义MyBatis中结果集处理器typeHandler
PS:本文基于MyBatis3.5.5版本源码。
概要
在MyBatis中,利用编程式进行数据查询,主要就是下面几行代码:
SqlSessionsession=sqlSessionFactory.openSession();UserMapperuserMapper=session.getMapper(UserMapper.class);ListLwUseruserList=userMapper.listUserByUserName("孤狼1号");
第一行是获取一个SqlSession对象在上一篇文章分析过了,想要详细了解的可以点击这里,第二行就是获取UserMapper接口,第三行一行代码就实现了整个查询语句的流程,接下来我们就来仔细分析一下第二和第三步。
获取Mapper接口(getMapper)
第二步是通过SqlSession对象是获取一个Mapper接口,这个流程还是相对简单的,下面就是我们调用session.getMapper方法之后的运行时序图:
1、在调用getMapper之后,会去Configuration对象中获取Mapper对象,因为在项目启动的时候就会把Mapper接口加载并解析存储到Configuration对象
图片
2、通过Configuration对象中的MapperRegistry对象属性,继续调用getMapper方法
图片
3、根据type类型,从MapperRegistry对象中的knownMappers获取到当前类型对应的代理工厂类,然后通过代理工厂类生成对应Mapper的代理类
图片
4、最终获取到我们接口对应的代理类MapperProxy对象
图片
而MapperProxy可以看到实现了InvocationHandler,使用的就是JDK动态代理。
图片
至此获取Mapper流程结束了,那么就有一个问题了MapperRegistry对象内的HashMap属性knownMappers中的数据是什么时候存进去的呢?
Mapper接口和映射文件是何时关联的
Mapper接口及其映射文件是在加载mybatis-config配置文件的时候存储进去的,下面就是时序图:
图片
1、首先我们会手动调用SqlSessionFactoryBuilder方法中的build()方法:
图片
2、然后会构造一个XMLConfigBuilder对象,并调用其parse方法:
img
3、然后会继续调用自己的parseConfiguration来解析配置文件,这里面就会分别去解析全局配置文件的顶级节点,其他的我们先不看,我们直接看最后解析mappers节点
img
4、继续调用自己的mapperElement来解析mappers文件(这个方法比较长,为了方便截图完整,所以把字体缩小了1号),可以看到,这里面分了四种方式来解析mappers节点的配置,对应了**「4种mapper配置方式」,而其中红框内的两种方式是直接配置的xml映射文件,蓝框内的两种方式是解析直接配置Mapper接口的方式,从这里也可以说明,「不论配置哪种方式,最终MyBatis都会将xml映射文件和Mapper接口进行关联」**。
img
5、我们先看第2种和第3中(直接配置xml映射文件的解析方式),会构建一个XMLMapperBuilder对象并调用其parse方法。
但是这里有一个问题,如果有多重继承或者多重依赖时在这里是可能会无法被完全解析的,比如说三个映射文件互相依赖,那么if里面(假设是最坏情况)只能加载1个,「失败2个」,然后走到下面if之外的代码又只能加载1个,「还有1个会失败」(如下代码中,只会处理1次,再次失败并不会继续加入in