介绍
该笔记是在学习拉勾教育 Java 高薪训练营后,结合课程和老师的视频,自己跟踪源码后做的笔记。
一级、二级缓存区别
- 存储结构, 一级缓存是存在内存 Map 中。二级缓存存储介质多样,可在内存、硬盘中,需要进行序列化和反序列化;
- 范围, 一级缓存是 sqlSession 级别的缓存,二级缓存是跨 SqlSession 的;
- 失效场景, 一级、二级缓存都是在执行插入、更新、删除时会失效,需要重新从数据库获取,避免脏读。另外一级缓存不能用于分布式场景,二级缓存需要使用 redis 来实现;
一级缓存
在开启一次数据库会话中,如果执行多次相同的查询 SQL,MyBatis 在第二次执行时会将从缓存中直接获取返回结果,而不再去数据库中查询。
在 openSession 中通过 configuration.newExecutor(tx, execType) 创建 Executor 对象。每个 SqlSession 都会创建一个 Executor,每个 Executor 都有一个一级缓存 LocalCache。
// 开启会话
SqlSession sqlSession = sqlSessionFactory.openSession()
Executor
Executor 接口的默认抽象实现类为 BaseExecutor,实现 Executor 接口的大部分功能。使用了模板方法设计模式,封装了一次操作的整个流程。
BaseExecutor#query
当用户发起查询时,根据执行的语句生成 MappedStatement,到一级缓存 localCache 中获取,缓存命名就直接返回结果给用户,没命中就查询数据库,将结果写入 localCache,最后返回结果给用户。
在如下代码中,localCache 为一个 HashMap,在构造函数 BaseExecutor 中会进行创建,对一级缓存的操作就是对 HashMap 的操作。如下为模板方法的查询流程,除了 doQuery 方法由子类实现,其余流程都由 BaseExecutor 实现封装。
- 先从一级缓存 localCache 获取,获取到则直接返回;
- queryFromDatabase(),获取不到时,会调用该方法从数据库中查询;
- queryFromDatabase#doQuery(),由继承的子类实现该抽象方法,进行数据库查询操作;
- localCache.putObject(),将数据库返回的结果存储到一级缓存中 localCache。
补充说明,这个 key 是一级缓存 HashMap 的 key,由 MappedStatement 的 Id、SQL 的 offset、SQL 的 limit、SQL 本身以及 SQL 中的参数 Params 构成的。
换句话说,判断是否为相同的 SQL 由这五个决定 Statement Id + Offset + Limmit + Sql + Params。
protected PerpetualCache localCache;
@SuppressWarnings("unchecked")
@Override
public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
// ...
try