前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【图文并茂】Mybatis执行SQL的4大基础组件详解

【图文并茂】Mybatis执行SQL的4大基础组件详解

作者头像
丁威
发布2019-06-11 14:55:44
9330
发布2019-06-11 14:55:44
举报
文章被收录于专栏:中间件兴趣圈中间件兴趣圈

温馨提示:本篇是源码分析Mybatis ShardingJdbc SQL语句执行的前置篇,重点阐述Mybatis执行SQL语句的4大核心组件。 源码分析Mybatis系列目录: 1、源码分析Mybatis MapperProxy初始化【图文并茂】 2、源码分析Mybatis MappedStatement的创建流程

Executor

sql执行器,其对应的类全路径:org.apache.ibatis.executor.Executor。

1.1 Executor类图

温馨提示:如果图片不清晰,可以浏览该网址:https://blog.csdn.net/prestigeding/article/details/90578125

  • Executor 执行器根据接口,定义update(更新或插入)、query(查询)、commit(提交事务)、rollback(回滚事务)。接下来简单介绍几个重要方法:
    • int update(MappedStatement ms, Object parameter) throws SQLException 更新或插入方法,其参数含义如下: 1、MappedStatement ms:SQL映射语句(Mapper.xml文件每一个方法对应一个MappedStatement对象) 2、Object parameter:参数,通常是List集合。
    • <E> List< E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) 查询方法,其参数含义如下: 1、RowBounds:行边界,主要值分页参数limit、offset。 2、ResultHandler resultHandler:结果处理器。
    • CacheKey createCacheKey(MappedStatement ms, Object param, RowBounds b, BoundSql s) 创建缓存Key,Mybatis一二级缓存的Key,可以看出Key由上述4个参数来决定。 1、BoundSql boundSql:可以通过该对象获取SQL语句。
  • CachingExecutor 支持结果缓存的SQL执行器,注意其设计模式的应用,该类中,会持有Executor的一个委托对象,CachingExecutor关注与缓存特定的逻辑,其最终的SQL执行由其委托对象来实现,即其内部的委托对象为BaseExecutor的实现类。
  • BaseExecutor Executor的基础实现类,该类为抽象类,关于查询、更新具体的实现由其子类来实现,下面4个都是其子类。
  • SimpleExecutor 简单的Executor执行器。
  • BatchExecutor 支持批量执行的Executor执行器。
  • ClosedExecutor 表示一个已关闭的Executor。
  • ReuseExecutor 支持重复使用Statement,以SQL为键,缓存Statement对象。
1.2 创建Executor

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

Configuration#newExecitor
代码语言:javascript
复制
 1public Executor newExecutor(Transaction transaction) {
 2  return newExecutor(transaction, defaultExecutorType);   // @1
 3}
 4
 5
 6public Executor newExecutor(Transaction transaction, ExecutorType executorType) {
 7  executorType = executorType == null ? defaultExecutorType : executorType;
 8  executorType = executorType == null ? ExecutorType.SIMPLE : executorType;
 9  Executor executor;
10  if (ExecutorType.BATCH == executorType) {   // @2
11    executor = new BatchExecutor(this, transaction);
12  } else if (ExecutorType.REUSE == executorType) {
13    executor = new ReuseExecutor(this, transaction);
14  } else {
15    executor = new SimpleExecutor(this, transaction);
16  }
17  if (cacheEnabled) { // @3
18    executor = new CachingExecutor(executor);
19  }
20  executor = (Executor) interceptorChain.pluginAll(executor);  // @4
21  return executor;
22}

从上面的代码可以看出,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的拆件机制,将在该系列文章中详细介绍。

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 c) 创建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的抽象实现类。我们来一一看一下其示例变量:
    • 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路由器,我们看一下其构造方法后,就会对该类了然于胸。
代码语言:javascript
复制
 1public RoutingStatementHandler(Executor executor, MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
 2
 3  switch (ms.getStatementType()) { // @1
 4    case STATEMENT:
 5      delegate = new SimpleStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
 6      break;
 7    case PREPARED:
 8      delegate = new PreparedStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
 9      break;
10    case CALLABLE:
11      delegate = new CallableStatementHandler(executor, ms, parameter, rowBounds, resultHandler, boundSql);
12      break;
13    default:
14      throw new ExecutorException("Unknown statement type: " + ms.getStatementType());
15  }
16
17}

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

2.2 创建StatementHandler
Configuration#newStatementHandler
代码语言:javascript
复制
1public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
2  StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql); // @1
3  statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler); // @2
4  return statementHandler;
5}

该方法的两个关键点如下: 代码@1:创建RoutingStatementHandler对象,在其内部再根据SQL语句的类型,创建对应的StatementHandler对象。 代码@2:对StatementHandler引入拆件机制,该部分将在该专题的后续文章中会详细介绍,这里暂时跳过。

ParameterHandler

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

3.1 ParameterHandler类图

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

3.2 创建ParameterHandler
Configuration#newParameterHandler
代码语言:javascript
复制
1public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {
2  ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement, parameterObject, boundSql);
3  parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler);  // @1
4  return parameterHandler;
5}

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

ResultSetHandler

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

4.1 ResultSetHandler类图

处理Jdbc ResultSet的处理器。

4.2 ResultSetHandler创建
Configuration#newResultSetHandler
代码语言:javascript
复制
1public ResultSetHandler newResultSetHandler(Executor executor, MappedStatement mappedStatement, RowBounds rowBounds, ParameterHandler parameterHandler,
2    ResultHandler resultHandler, BoundSql boundSql) {
3  ResultSetHandler resultSetHandler = new DefaultResultSetHandler(executor, mappedStatement, parameterHandler, resultHandler, boundSql, rowBounds);
4  resultSetHandler = (ResultSetHandler) interceptorChain.pluginAll(resultSetHandler);
5  return resultSetHandler;
6}

同样支持插件化机制,我们也稍微再看一下其调用链:

可以看出其调用的入口为SQL执行时。

本文作为下一篇《源码分析Mybatis整合ShardingJdbc SQL执行流程》的前置篇,详细介绍Executor、StatementHandler、ParameterHandler、ResultSetHandler的具体职责,以类图为基础详细介绍其核心方法的作用,然后讲述这些对象如何创建并引出Mybatis拆件机制。并引出Mybatis拆件机制。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-05-26,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 中间件兴趣圈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Executor
    • 1.1 Executor类图
      • 1.2 创建Executor
        • Configuration#newExecitor
        • StatementHandler
          • 2.1 StatementHandler类图
            • 2.2 创建StatementHandler
              • Configuration#newStatementHandler
              • ParameterHandler
                • 3.1 ParameterHandler类图
                  • 3.2 创建ParameterHandler
                    • Configuration#newParameterHandler
                    • ResultSetHandler
                      • 4.1 ResultSetHandler类图
                        • 4.2 ResultSetHandler创建
                          • Configuration#newResultSetHandler
                          领券
                          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档