MyBatis 的一、二级缓存的区别及源码分析

本文介绍了MyBatis的一级和二级缓存的区别,包括存储结构、范围和失效场景。一级缓存存在于SqlSession级别,而二级缓存可以跨SqlSession并可扩展到分布式场景。文章详细分析了Executor的查询和更新流程,并指出在分布式环境下,使用Redis等集中式缓存来保证数据安全的重要性。
摘要由CSDN通过智能技术生成

介绍

该笔记是在学习拉勾教育 Java 高薪训练营后,结合课程和老师的视频,自己跟踪源码后做的笔记。

一级、二级缓存区别

  • 存储结构, 一级缓存是存在内存 Map 中。二级缓存存储介质多样,可在内存、硬盘中,需要进行序列化和反序列化;
  • 范围, 一级缓存是 sqlSession 级别的缓存,二级缓存是跨 SqlSession 的;
  • 失效场景, 一级、二级缓存都是在执行插入、更新、删除时会失效,需要重新从数据库获取,避免脏读。另外一级缓存不能用于分布式场景,二级缓存需要使用 redis 来实现;

一级缓存

在开启一次数据库会话中,如果执行多次相同的查询 SQL,MyBatis 在第二次执行时会将从缓存中直接获取返回结果,而不再去数据库中查询。
  在 openSession 中通过 configuration.newExecutor(tx, execType) 创建 Executor 对象。每个 SqlSession 都会创建一个 Executor,每个 Executor 都有一个一级缓存 LocalCache。

// 开启会话
SqlSession sqlSession = sqlSessionFactory.openSession()

图片来自聊聊 MyBatis 缓存机制

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 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值