Mybatis中SqlSession执行Mapper的过程

上篇文章介绍了Mybatis中的SqlSession组件的创建,创建SqlSession之后需要利用SqlSession执行对应的Mapper。Mapper分为两部分,一部分是JAVA的Mapper接口,另一部分是Mapper.xml或者注解的Sql语句。本篇文章分析SqlSession是如何执行Mapper的。

Mapper接口动态代理对象的生成

获取SqlSession之后我们需要通过SqlSession的getMapper()方法,传入当前Mapper接口的Class对象。返回值是一个绑定了当前SqlSession的Mapper对象,之后便可以调用该Mapper对象定义好的接口方法。如下所示:

这种通过调用Mapper接口定义的方法也是写代码时使用Myabtis的方式。那么大家有没有思考过一个问题,平常能直接调用接口定义的方法吗?

也就是说肯定有一个对应的实现类,创建该实现类的实例之后才能通过该类的实例调用该方法。

我们可以看看session.getMapper()方法:

如上图所示,SqlSession.getMapper()通过configuration的getMapper()方法获取对应的实例。Configuration中getMapper()的实现为:

如上图所示,通过mapperRegistry的getMapper()方法获取对应的实例。MapperRegistry最终从kownMappers中获取对应的MapperProxyFactory,最终通过MapperProxyFactory的newInstance()方法创建对应的MapperProxy实例,其代码如下:

上述代码中需要搞清楚MapperProxyFactory从何而来。

configuration中的getMapper()方法里从mapperRegistry中获取Mapper的代理对象,mapperRegistry的定义为

其声明了一个addMapper()方法,通过addMapper()方法将对应的MapperProxyFactory放入knowMappers中

而addMapper()方法在初始化configuration时解析mappers标签相关属性后被调用。

代理对象的invoke方法

生成了动态代理对象之后,调用动态代理对象的方法时会执行该动态代理对象的invoke方法,MapperProxy的invoke()方法如下所示,最终调用PlainMethodInvoker的invoke()方法。

PlainMethodInvoker的invoke()方法如下所示:

最终调用MapperMethod的execute()方法。

execute()方法最终通过SqlSession完成与Myabtis核心组件的交互。

SqlSession执行Mapper

Mybatis调用MapperMethod的execute()方法,在该方法里用SqlSession提供的api执行数据库的操作。 接下来以insert()方法举例,看看SqlSession如何执行Mapper。

构造MapperMethod需要两个属性,一是commond,二是method。这两个属性在MapperMehtod的构造函数中创建,如下所示:

SqlCommand对象用于获取SQL语句的类型、Mapper的Id等信息;MethodSignature对象用于获取方法的签名信息,例如Mapper方法的参数名、参数注解等信息。

构造完成之后调用SqlSession的insert操作,最终转变为调用SqlSession的update操作,如下所示:

如上图所示,在update()方法中先获取MappedStatement对象,之后调用SqlSession持有的executor的update()方法。

前面文章中介绍过MappedStatement对象用来描述SQL相关信息。SQL信息分为两种,一是mapper.xml配置的SQL信息,另一类是通过注解配置的SQL信息,其在configuration初始化时被加载。

之后调用sqlSession持有的executor的update()方法。以SimpleExecutor为例,调用update()之后会调用到SimpleExecutor的doUpdate()方法。

如上所示,先获取到configuration对象,之后获取到对应不同类型的StatementHandler。之后通过调用prepareStatement()获取Connection对象的Statement对象。

随后调用handler的update()方法完成此次交互。以SimpleStatementHandler为例,其update的实现如下所示:

调用JDBC的statement对象的execute方法完成与JDBC的交互,至此MyBatis执行Mapper完毕。

总结

Myabatis先获取Mapper接口对应的动态代理对象,执行其invoke()方法,最终调用MapperMethod的execute()方法,execute方法通过SqlSession完成与Mybatis核心组件的交互。 SqlSession持有executor,通过调用executor封装的方法完成与JDBC的交互。

相关推荐
没有bug.的程序员4 小时前
JAVA面试宝典 - 《MyBatis 进阶:插件开发与二级缓存》
java·面试·mybatis
码不停蹄的玄黓10 小时前
深入理解MyBatis延迟加载:原理、配置与实战优化
java·mybatis·延迟加载
hello早上好15 小时前
JPA、缓存、数据源与连接池、简介
java·mybatis
梁辰兴15 小时前
企业培训笔记:宠物信息管理--实现宠物信息分页查询
笔记·elementui·mybatis·vue3·springboot·宠物
用户38022585982417 小时前
vue3源码解析:watch的实现
前端·vue.js·源码
鲁子狄17 小时前
[笔记] 动态 SQL 查询技术解析:构建灵活高效的企业级数据访问层
java·spring boot·笔记·sql·mysql·mybatis
plusone18 小时前
【React18源码解析】(零)写在最开始
react.js·前端框架·源码
拼搏@1 天前
第十六天,7月10日,八股
java·mybatis
军军君011 天前
基于Springboot+UniApp+Ai实现模拟面试小工具四:后端项目基础框架搭建下
spring boot·spring·面试·elementui·typescript·uni-app·mybatis
荔枝荔枝荔枝1 天前
【Vue源码学习】Vue新手友好!为什么vue2 this能够直接获取到data和methods中的属性?
vue.js·源码