Mybatis执行SQL的4大基础组件详解

在Mybatis中,Executor的创建由Configuration对象来创建,具体的代码如下:

Configuration#newExecitor

public Executor newExecutor(Transaction transaction) {

return newExecutor(transaction, defaultExecutorType); // @1

}

public Executor newExecutor(Transaction transaction, ExecutorType executorType) {

executorType = executorType == null ? defaultExecutorType : executorType;

executorType = executorType == null ? ExecutorType.SIMPLE : executorType;

Executor executor;

if (ExecutorType.BATCH == executorType) { // @2

executor = new BatchExecutor(this, transaction);

} else if (ExecutorType.REUSE == executorType) {

executor = new ReuseExecutor(this, transaction);

} else {

executor = new SimpleExecutor(this, transaction);

}

if (cacheEnabled) { // @3

executor = new CachingExecutor(executor);

}

executor = (Executor) interceptorChain.pluginAll(executor); // @4

return executor;

}

从上面的代码可以看出,Executor的创建由如下三个关键点:

代码@1:默认的ExecutorType为ExecutorType.SIMPLE,即默认创建的Executory为SimpleExecutor。

代码@2:根据executorType的值创建对应的Executory。

代码@3:如果cacheEnabled为true,则创建CachingExecutory,然后在其内部持有上面创建的Executor,cacheEnabled默认为true,则默认创建的Executor为CachingExecutor,并且其内部包裹着SimpleExecutor。

代码@4:使用InterceptorChain.pluginAll为executor创建代理对象,即Mybatis的拆件机制,将在该系列文章中详细介绍。

2、StatementHandler


在学习StatementHandler之前,我们先来回顾一下JDBC相关的知识。JDBC与语句执行的两大主流对象:java.sql.Statement、java.sql.PrepareStatement对象大家应该不会陌生,该对象的execute方法就是执行SQL语句的入口,通过java.sql.Connection对象创建Statement对象。Mybatis的StatementHandler,是Mybatis创建Statement对象的处理器,即StatementHandler会接管Statement对象的创建。

2.1 StatementHandler类图

在这里插入图片描述

  • StatementHandler

根接口,我们重点关注一下其定义的方法:

  • Statement prepare(Connection connection)

创建Statement对象,即该方法会通过Connection对象创建Statement对象。

  • void parameterize(Statement statement)

对Statement对象参数化,特别是PreapreStatement对象。

  • void batch(Statement statement)

批量执行SQL。

  • int update(Statement statement)

更新操作。

  • < E> List< E> query(Statement statement, ResultHandler resultHandler)

查询操作。

  • BoundSql getBoundSql()

获取SQL语句。

  • ParameterHandler getParameterHandler()

获取对应的参数处理器。

  • BaseStatementHandler

StatementHandler的抽象实现类,SimpleStatementHandler、PrepareStatementHandler、CallableStatementHandler是其子类。

我们来一一看一下其示例变量:

  • Configuration configuration

Mybatis全局配置对象。

  • ObjectFactory objectFactory

对象工厂。

  • TypeHandlerRegistry typeHandlerRegistry

类型注册器。

  • ResultSetHandler resultSetHandler

结果集Handler。

  • ParameterHandler parameterHandler

参数处理器Handler。

  • Executor executor

SQL执行器。

  • MappedStatement mappedStatement

SQL映射语句(Mapper.xml文件每一个方法对应一个MappedStatement对象)

  • RowBounds rowBounds

行边界,主要值分页参数limit、offset。

  • BoundSql boundSql

可以通过该对象获取SQL语句。

  • SimpleStatementHandler

具体的StatementHandler实现器,java.sql.Statement对象创建处理器。

  • PrepareStatementHandler

java.sql.PrepareStatement对象的创建处理器。

  • CallableStatementHandler

java.sql.CallableStatement对象的创建处理器,可用来执行存储过程调用的Statement。

  • RoutingStatementHandler

StatementHandler路由器,我们看一下其构造方法后,就会对该类了然于胸。

public RoutingStatementHandler(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {

switch (ms.getStatementType()) { // @1

case STATEMENT:

delegate = new SimpleStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);

break;

case PREPARED:

delegate = new PreparedStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);

break;

case CALLABLE:

delegate = new CallableStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);

break;

default:

throw new ExecutorException("Unknown statement type: " + ms.getStatementType());

}

}

原来是会根据MappedStatement对象的statementType创建对应的StatementHandler。

2.2 创建StatementHandler

Configuration#newStatementHandler

public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {

StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql); // @1

statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler); // @2

return statementHandler;

}

该方法的两个关键点如下:

代码@1:创建RoutingStatementHandler对象,在其内部再根据SQL语句的类型,创建对应的StatementHandler对象。

代码@2:对StatementHandler引入拆件机制,该部分将在该专题的后续文章中会详细介绍,这里暂时跳过。

3、ParameterHandler


参数处理器。同样我们先来看一下其类图。

3.1 ParameterHandler类图

在这里插入图片描述

这个比较简单,就是处理PreparedStatemet接口的参数化处理,也可以顺便看一下其调用链(该部分会在下一篇中详细介绍)。

在这里插入图片描述

3.2 创建ParameterHandler

Configuration#newParameterHandler

public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {

ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement, parameterObject, boundSql);

parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler); // @1

return parameterHandler;

}

同样该接口也支持插件化机制。

4、ResultSetHandler


处理结果的Handler。我们同样看一下其类图。

4.1 ResultSetHandler类图

在这里插入图片描述

处理Jdbc ResultSet的处理器。

4.2 ResultSetHandler创建

Configuration#newResultSetHandler

public ResultSetHandler newResultSetHandler(Executor executor, MappedStatement mappedStatement, RowBounds rowBounds, ParameterHandler parameterHandler,

ResultHandler resultHandler, BoundSql boundSql) {

ResultSetHandler resultSetHandler = new DefaultResultSetHandler(executor, mappedStatement, parameterHandler, resultHandler, boundSql, rowBounds);

resultSetHandler = (ResultSetHandler) interceptorChain.pluginAll(resultSetHandler);

return resultSetHandler;

}

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
8h9D-1715840761475)]

[外链图片转存中…(img-1Pb7kxbn-1715840761476)]

[外链图片转存中…(img-rqD39qmu-1715840761476)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值