Executor 执行器

今天分享一下 Executor。它在框架中是具体sql的执行器,sqlSession(门面模式)封装通用的api,把具体操作委派给 Executor 执行,Executor协同BoundSql,StatementHandler,ParameterHandler 和 ResultSetHandler 完成工作。

它使用装饰器的方式组织 Executor 对象。如 CachingExecutor 装饰了SimpleExecutor 提供二级缓存功能。

可以通过插件机制扩展功能。mybatisplus 就是通过插件机制扩展的功能。

下面是更新流程,Executor 处于流程中间蓝色部分,缓存执行器,基础执行器,简单执行器三个 Executor 通过责任链的方式组织起来,各司其职,一同完成执行工作。,可以感受到它的作用是承上启下。

执行器介绍

Mybatis 一共提供了四种执行器的实现和一个模板类:

  • 基础执行器 BaseExecutor:实现Executor接口的抽象类,实现了框架逻辑,具体的逻辑委派给子类实现。一级缓存也是在这里实现的。
  • 缓存执行器 CachingExecutor:实现了二级缓存,是jvm级别的全局缓存。
  • 简单执行器 SimpleExecutor:继承自 BaseExecutor,具体执行逻辑的实现。
  • 重用执行器 ReuseExecutor:相同的 sql 只会预编译一次。
  • 批处理执行器 BatchExecutor:批处理执行器 使用 JDBC 的batch API 执行 SQL 的批量操作,如insert 或者 update。select的逻辑和 SimpleExecutor 的实现一样。

今天介绍 SimpleExecutor,ReuseExecutor 和 BatchExecutor 三个执行器的特定和逻辑, CachingExecutor 的功能是提供二级缓存,暂时不在这里介绍。

SimpleExecutor

简单执行器顾名思义,处理的逻辑比较简单直接,来一个 sql 预编译一个,处理一个。 示例代码如下:

// 创建 SimpleExecutor SimpleExecutor simpleExecutor = new SimpleExecutor(sessionFactory.getConfiguration(),jdbcTransaction);// 获取 MappedStatement final MappedStatement ms = sessionFactory.getConfiguration().getMappedStatement("example.mapper.UserMapper.getUserByID");final BoundSql boundSql = ms.getBoundSql(1);// 执行 2 次查询simpleExecutor.doQuery(ms, 1, RowBounds.DEFAULT, Executor.NO_RESULT_HANDLER, boundSql);simpleExecutor.doQuery(ms, 1, RowBounds.DEFAULT, Executor.NO_RESULT_HANDLER, boundSql);

执行结果:

[DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: select * from `user` where id = ? [DEBUG][main] m.p.ThresholdInterceptor.intercept ThresholdInterceptor plugin... [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: 1(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug <==      Total: 1 [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: select * from `user` where id = ? [DEBUG][main] m.p.ThresholdInterceptor.intercept ThresholdInterceptor plugin... [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: 1(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug <==      Total: 1 

通过日志看到,虽然执行相同的 sql 但是每次都要执行预编译。这是一个需要优化的点。

ReuseExecutor

ReuseExecutor 对相同 SQL 重复编译做了优化,相同的 sql 的 Statement 只创建一个。

示例代码上面一样,只是把 SimpleExecutor 换成 ReuseExecutor 。 从执行我们看到,Preparing 只有一次,执行结果也是正确的:

[DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: select * from `user` where id = ? [DEBUG][main] m.p.ThresholdInterceptor.intercept ThresholdInterceptor plugin... [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: 1(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug <==      Total: 1 [DEBUG][main] m.p.ThresholdInterceptor.intercept ThresholdInterceptor plugin... [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: 1(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug <==      Total: 1 

他是怎么做到的呢?翻开代码看看实现,其实逻辑也很简单,用 SQL 当作 key 保存对应的 Statement 来实现重用。

  private Statement prepareStatement(StatementHandler handler, Log statementLog) throws SQLException {    Statement stmt;    BoundSql boundSql = handler.getBoundSql();    String sql = boundSql.getSql();    // 关键逻辑,通过 sql 判断是否已经创建了 Statement,如果有则重用。    if (hasStatementFor(sql)) {      stmt = getStatement(sql);      applyTransactionTimeout(stmt);    } else {      Connection connection = getConnection(statementLog);      stmt = handler.prepare(connection, transaction.getTimeout());      putStatement(sql, stmt);    }    handler.parameterize(stmt);    return stmt;  }  private final Map statementMap = new HashMap<>();  private boolean hasStatementFor(String sql) {    try {      Statement statement = statementMap.get(sql);      return statement != null && !statement.getConnection().isClosed();    } catch (SQLException e) {      return false;    }  }

BatchExecutor

有些场景下,我们要批量保存或者删除,更新数据,这时候我们一条一条的执行效率就会很低,需要一个批量执行的机制。

JDBC 批量操作

批量操作可以把相关的sql打包成一个 batch,一次发送到服务器,减少和服务器的交互,也就是 RTT 时间。

使用批量操作前要确认服务器是否支持批量操作,可通过 DatabaseMetaData.supportsBatchUpdates() 方法的返回值来判断。

实例代码,通过 JDBC 提供的 API 执行批量操作。

Connection conn = DriverManager.getConnection(DB_URL, USER, PASS);DatabaseMetaData metaData = conn.getMetaData();System.out.println("metaData.supportsBatchUpdates() = " + metaData.supportsBatchUpdates());//执行 sqlSystem.out.println("Creating statement...");String sql = "update user set name=? where id = ?";pstmt = conn.prepareStatement(sql);// 设置变量pstmt.setString(1, "Pappu");pstmt.setInt(2, 1);// 添加到 batchpstmt.addBatch();// 设置变量pstmt.setString(1, "Pawan");pstmt.setInt(2, 2);// 添加到 batchpstmt.addBatch();//执行,并获取结果int[] count = pstmt.executeBatch();

Mybatis 如何实现

Mybatis 只有对 update 有支持批量操作,并且需要手动 flushStatements。

insert、delete、update,都是update操作

    BatchExecutor batchExecutor = new BatchExecutor(configuration, jdbcTransaction);    final MappedStatement update = configuration        .getMappedStatement("dm.UserMapper.updateName");    final MappedStatement delete = configuration        .getMappedStatement("dm.UserMapper.deleteById");    final MappedStatement get = sessionFactory.getConfiguration()        .getMappedStatement("dm.UserMapper.getUserByID");    final MappedStatement insertUser = sessionFactory.getConfiguration()        .getMappedStatement("dm.UserMapper.insertUser");    // query    batchExecutor.doUpdate(insertUser, new User().setName("" + new Date()));    batchExecutor.doUpdate(insertUser, new User().setName("" + new Date()));    // batch update    User user = new User();    user.setId(2);    user.setName("" + new Date());    batchExecutor.doUpdate(update, user);    user.setId(3);    batchExecutor.doUpdate(update, user);    batchExecutor.doUpdate(insertUser, new User().setName("" + new Date()));    //    final List batchResults = batchExecutor.flushStatements(false);    jdbcTransaction.commit();    printBatchResult(batchResults);

执行日志:

[DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: insert into `user` (name) values(?); [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: Sat Jul 04 15:07:30 CST 2020(String) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: Sat Jul 04 15:07:30 CST 2020(String) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: update `user` set name=? where id = ? [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: Sat Jul 04 15:07:30 CST 2020(String), 2(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: Sat Jul 04 15:07:30 CST 2020(String), 3(Integer) [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==>  Preparing: insert into `user` (name) values(?); [DEBUG][main] o.a.i.l.j.BaseJdbcLogger.debug ==> Parameters: Sat Jul 04 15:07:30 CST 2020(String) [DEBUG][main] o.a.i.t.j.JdbcTransaction.commit Committing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@4b3ed2f0] 第 1 个结果[1, 1]第 2 个结果[1, 1]第 3 个结果[1]

从日志可以看到看到清晰的执行过程。

  • 第一个insert语句后面跟着两个参数,是一个statement。对应第 1 个结果
  • 第二个update语句后面跟着两个参数,是一个statement。对应第 2 个结果
  • 第三个insert语句后面跟着两个参数,是一个statement。对应第 3 个结果

整体逻辑和程序是一致的,但是有个问题,为什么三个相同的 insert,会分开两个结果返回呢?

这是因为 Mybatis 为了保证批次和逻辑顺序一致做了优化,并不是相同的sql就放到相同的statement。而是要按照执行顺序把相同的sql当作一个批次。

从代码中可以看到这部分逻辑:

public int doUpdate(MappedStatement ms, Object parameterObject) throws SQLException {  if (sql.equals(currentSql) && ms.equals(currentStatement)) {    使用当前的 statement  } else {    创建新的statement  }}

总结

网络上有些文章介绍使用 foreach 的方式执行批量操作,我个人不建议这样操作。

  1. 因为 JDBC 已经提供了批量操作的接口,符合规范,兼容性和性能更好。
  2. foreach拼接的 sql 比较长,会增加网络流量,而且驱动对sql长度是有限制的,并且要增加allowMultiQueries参数。
  3. foreach 拼接的 sql 每次都不一定相同,服务器会重新编译。

Mysql 的 sql 执行流程是连接器,查询缓存,分析器,优化器,执行器。分析器先会做“词法分析”。优化器是在表里面有多个索引的时候,决定使用哪个索引;或者在一个语句有多表关联(join)的时候,决定各个表的连接顺序。在适合的场景使用 ReuseExecutor 或 BatchExecutor 不仅可以提高性能,还可以减少对 Mysql 服务器的压力。

作者:但莫
链接:https://juejin.im/post/6854573210420772877
来源:掘金

mybatis 执行sql_来,全搞懂,原来Mybatis执行一个sql有这么多类型,绝相关推荐

  1. 纯干货!15000 字语法手册分享给你,看完搞懂,再也不担心SQL写不好了

    点击上方"民工哥技术之路",选择"设为星标" 回复"1024"获取独家整理的学习资料! 一.基础 1.创建数据库 CREATE DATABA ...

  2. 一文搞懂从浏览器输入一个URL到页面出现都经历了哪些过程

    1 过程一览 DNS解析(如果IP直接访问则此过程省略) 客户端与服务端进行TCP三次握手连接 客户端发送HTTP请求 服务器处理请求并返回HTTP报文 浏览器解析渲染页面 连接结束 2 细节剖析 2 ...

  3. 程序人生 - 只要看这一篇,车险全搞懂(值得收藏)

    现在只要不是条件太差的家庭,基本上每家都有一辆车,条件好的甚至是人手一台. 俗话说,常在河边走,哪有不湿鞋的,有了私家车,那就伴随着必须买保险.遇到交通事故,有车险护身,绝对是一个必不可少的保障. 车 ...

  4. webpack5和webpack4详解(一篇全搞懂,值得收藏)

    文章目录 本篇的目的 1.webpack基础 1.1.webpack开发环境搭建 1.2.webpack配置核心 1.3.开发及生产环境分离 1.4.处理静态资源 2.webpack热门插件 2.1. ...

  5. 一篇文章全搞懂!B2B

    作为一名刚入行不久的电商新人,时常被各种专业名词搞的晕头转向,公司开会或者行业交流时候没听懂也只能假装呵呵一笑(话说这种呵呵一笑是什么态度!摔).痛心疾首,小编决定总结近期所有我听过的没听过的互联网专 ...

  6. mybatis直接执行sql_拼多多二面:Mybatis是如何执行一条SQL命令的?

    Mybatis中的Sql命令,在枚举类SqlCommandType中定义的. public enum SqlCommandType { UNKNOWN, INSERT, UPDATE, DELETE, ...

  7. 服务注册与发现-全搞懂

    关注公众号[高性能架构探索],第一时间分享技术干货.回复[pdf],免费获取计算机必备经典书籍 最近一直想写这个话题,也一直在构思,但不知道从何入手,或者说不知道写哪方面.如果单纯写如何实现,这个未免 ...

  8. RPC基础之负载均衡算法一文全搞懂

    文章目录 前言 一.轮询法 二.随机法 三.地址哈希法 四.权重法 五.最小连接数法 六.拓展 结语 前言 在微服务的技术体系中,服务消费者从配置中心获取到服务的地址后,需要选择其中的一台服务器发起R ...

  9. 查看正在执行的事务_看懂sql_trace--分析执行计划及CBO行为

    概述 Oracle数据库排查问题.实验测试.优化的时候一般都会用到trace文件来分析,这里面就涉及到sql_trace跟10046事件了,下面分别做一下介绍. SQL_TRACE SQL_TRACE ...

最新文章

  1. webpack与babel的深奥,渣渣的我只能做个小笔记(持续更新)
  2. 面试中遇到这 3 个SQL问题,最容易掉坑里!
  3. Ubuntu 查看磁盘空间 及目录容量
  4. Windows MinGW cmake 安装编译Opencv 3.4.3 C++开发环境
  5. Javascript闭包简单理解
  6. 小型功率放大器的设计与制作——功率放大器的设计方法
  7. 英特尔第十代处理器为什么不支持win7_为什么i7处理器不支持win7系统,背后的真实原因?...
  8. 堪比熊猫烧香!中国新型蠕虫病毒大爆发!电脑瞬间报废
  9. 转义字符 \r \v \f
  10. C语言每日一练——第85天:三色球问题
  11. 拼多多引流大法,日引百人太简单
  12. 爬虫mysql数据清洗_爬虫分析之数据存储——基于MySQL,Scrapy
  13. HTML5期末大作业:订餐系统网站设计——绿色的网上订餐系统(23个页面) 网页作品 订餐系统网页设计作业模板 学生网页制作源代码下载
  14. 菏泽一中2021级高考成绩查询入口,2021年菏泽中考各学校录取分数线,历年菏泽中考分数线汇总...
  15. 聊一下“深度学习”的基本思想
  16. margin-left:-100%
  17. 从2.3.3到4.1.1:最全的android系统源码下载大集合
  18. 宏观经济学_宏观经济指标
  19. 云台山最具特色的景点——“长宝第一洞天”龙泉洞
  20. A4 纸张上打印预览

热门文章

  1. 图像处理函数——roicolor、regionprops、padarray、nlfilter
  2. matlab 未定义mat2gray,图像检测mat2gray出错,,,求解小白给跪
  3. SoC EDS 17.0 和 DS-5 下载和安装
  4. mysql 高并发 集群架构_一种高并发的GPU集群架构及其负载均衡方法技术
  5. Qcon演讲实录|手机淘宝客户端的攻防演练实践
  6. apache支持laravel路由_swoole运行模式加速laravel应用的详细介绍
  7. expo的未来,超乎你想象
  8. Oracle数据库练习题
  9. Dom4j解析kml (字符串,文件两种方式)
  10. 如何用Graphpad prism绘制点图(Dot plot) ?