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

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

Executor

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

1.1 Executor类图

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

舒适提示:若是图片不清晰,能够浏览该网址:http://www.javashuo.com/article/p-uiuwcfeb-hb.htmlapache

  • Executor
    执行器根据接口,定义update(更新或插入)、query(查询)、commit(提交事务)、rollback(回滚事务)。接下来简单介绍几个重要方法:设计模式

  • int update(MappedStatement ms, Object parameter) throws SQLException
    更新或插入方法,其参数含义以下:
    一、MappedStatement ms:SQL映射语句(Mapper.xml文件每个方法对应一个MappedStatement对象)
    二、Object parameter:参数,一般是List集合。
  • <E> List< E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler)
    查询方法,其参数含义以下:
    一、RowBounds:行边界,主要值分页参数limit、offset。
    二、ResultHandler resultHandler:结果处理器。
    CacheKey createCacheKey(MappedStatement ms, Object param, RowBounds b, BoundSql s)
    建立缓存Key,Mybatis一二级缓存的Key,能够看出Key由上述4个参数来决定。
    一、BoundSql boundSql:能够经过该对象获取SQL语句。
  • CachingExecutor
    支持结果缓存的SQL执行器,注意其设计模式的应用,该类中,会持有Executor的一个委托对象,CachingExecutor关注与缓存特定的逻辑,其最终的SQL执行由其委托对象来实现,即其内部的委托对象为BaseExecutor的实现类。缓存

  • BaseExecutor
    Executor的基础实现类,该类为抽象类,关于查询、更新具体的实现由其子类来实现,下面4个都是其子类。微信

  • SimpleExecutor
    简单的Executor执行器。app

  • BatchExecutor
    支持批量执行的Executor执行器。运维

  • ClosedExecutor
    表示一个已关闭的Executor。ide

  • ReuseExecutor
    支持重复使用Statement,以SQL为键,缓存Statement对象。

1.2 建立Executor

在Mybatis中,Executor的建立由Configuration对象来建立,具体的代码以下:源码分析

Configuration#newExecitor

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类图

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

  • 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路由器,咱们看一下其构造方法后,就会对该类了然于胸。

    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

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类图

【图文并茂】Mybatis执行SQL的4大基础组件详解
这个比较简单,就是处理PreparedStatemet接口的参数化处理,也能够顺便看一下其调用链(该部分会在下一篇中详细介绍)。
【图文并茂】Mybatis执行SQL的4大基础组件详解

3.2 建立ParameterHandler

Configuration#newParameterHandler

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类图

【图文并茂】Mybatis执行SQL的4大基础组件详解
处理Jdbc ResultSet的处理器。

4.2 ResultSetHandler建立

Configuration#newResultSetHandler

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}

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

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

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

更多文章请关注微信公众号:

【图文并茂】Mybatis执行SQL的4大基础组件详解
一波广告来袭,做者新书《RocketMQ技术内幕》已出版上市:

【图文并茂】Mybatis执行SQL的4大基础组件详解《RocketMQ技术内幕》已出版上市,目前可在主流购物平台(京东、天猫等)购买,本书从源码角度深度分析了RocketMQ NameServer、消息发送、消息存储、消息消费、消息过滤、主从同步HA、事务消息;在实战篇重点介绍了RocketMQ运维管理界面与当前支持的39个运维命令;并在附录部分罗列了RocketMQ几乎全部的配置参数。本书获得了RocketMQ创始人、阿里巴巴Messaging开源技术负责人、Linux OpenMessaging 主席的高度承认并做序推荐。目前是国内第一本成体系剖析RocketMQ的书籍。

相关文章
相关标签/搜索