一 执行器层次结构
mybatis 大体上的执行器关系链路如图。上层的 CachingExecutor 和 BaseExecutor 各自处理一级缓存。
最基础的三个执行器负责与数据库交互。并且三者之间各有区别。
1.1 SimpleExecutor
最基础的简单执行器
org.apache.ibatis.executor.SimpleExecutor
当我们使用 mybatis 封装好的 sqlSession 会话来进行数据库交互的时候,debug 代码,就会发现底层的调用,默认就是执行到了
org.apache.ibatis.executor.SimpleExecutor#doQuery
一次查询操作在基础执行器中的流程逻辑是:
- 根据 mybatis 配置的 sql 全局 id 和 sql 语句的映射实体对象 mappedStatement 来获取一个配置类实体。这个配置类包含了环境配置/指定执行器类型/日志配置等等
- 创建一个 statement 处理器,包含了语句映射关系/分页参数/前面创建的配置类引用等等
- 预编译 sql 语句
- 执行查询操作
每次调用简单执行器的查询动作,都会执行完整的 sql 预编译过程,即这个方法:
org.apache.ibatis.executor.SimpleExecutor#prepareStatement
这是一个细微的性能损耗优化点。使用另一个基类执行器 ReuseExecutor 可以避免重复预编译完全一样的 sql,当然这个优化范围也是在 同一个 session 域内。
1.2 ReuseExecutor
复用执行器
org.apache.ibatis.executor.ReuseExecutor
这里的 复用 指的是对 sql 语句预编译动作的复用。执行器内会缓存预编译通过的 sql,并在接到相同的 sql 和执行请求的时候,从本地缓存中检索出之前已经执行过的内容并执行。这样就可以省略预编译过程的性能损耗。
本地缓存,是一个 hashmap 结构:
private final Map<String, Statement> statementMap = new HashMap<>();
保存的键值对就是 sql-stmt 的映射关系集合。
mybatis 默认使用的底层执行器是 SimpleExecutor,如果想要使用 ReuseExecutor,需要在配置文件中新增配置,指定使用的执行器。
1.3 BatchExecutor
批处理执行器
org.apache.ibatis.executor.BatchExecutor
用于一次性操作更新大量数据的场景。批处理执行器在用户查询场景的时候,预编译和 simpleExecutor 是没有差别的。
注意批处理操作必须手动处理/提交事物。
网友评论