Mybatis缓存的生命周期、使用的特殊情况

以下场景均在Spring Boot程序中,并非手动创建SqlSession使用。

在回答这个问题之前,我们先来回顾一下,Mybatis的一级二级缓存是啥。

一级二级缓存

是什么

  1. 一级缓存(本地缓存):一级缓存是SqlSession级别的缓存,当我们执行查询时,MyBatis会先将查询结果放在当前SqlSession的缓存中,如果后续有相同的查询,MyBatis会直接从缓存中取出结果,而不会再次执行 SQL 查询。
  2. 二级缓存(全局缓存):二级缓存是 Mapper 级别的缓存,它可以被多个SqlSession共享。当一个SqlSession 查询数据时,MyBatis会先在一级缓存中查找,如果没有找到,再去二级缓存中查找,如果还没有找到,最后才会去数据库查询。
缓存类型 生命周期 如何开启 位置
一级缓存 随SqlSession的生命周期 默认启用,无法关闭 BaseExecutorlocalCache参数中
二级缓存 跟随程序的生命周期 xml文件标识<cache/>或者使用@CacheNamespace注解 MappedStatement中的cache参数

问题

回到问题本身,其实就是一个缓存生命周期的回答,是否每次都会使用呢?

并不是,得看是如何使用的。有两种情况:

  1. 显式使用SqlSession:在查询后,没有显式调用clearCache并且在同一个上下文二次查询后,缓存使用。
java 复制代码
try (SqlSession session = sqlSessionFactory.openSession()) {
    // 第一次查询,将结果放入缓存
    session.select();
    
    // 第二次查询,直接使用上一次的搜索结果
    session.select();
    // 清除缓存
    session.clearCache();
}
  1. 由Spring容器管理的SqlSession:在Spring中,每次调用Mapper的方法都会创建一个新的SqlSession,自然而然也不会使用缓存了。当然,开启了事务另说。

题外话

有没有什么额外回答能在回答的时候加一下分呢?

那么我们就要想想在什么场景下,使用了上一次的搜索结果会有问题呢?

没错就是它:数据库的事务隔离级别

在MySQL中不同的事务隔离级别 对查询起着不同的影响

其中在可重复读(REPEATABLE READ)以下的级别都会受到Mybatis 一级二级缓存的影响。那么为什么呢?

读未提交(READ UNCOMMITTED)、读已提交(READ COMMITTED)都可以在事务查询时获取到其他事务的结果,所以如果使用了Mybatis的缓存 的话,就跟数据库的查询结果不一致了。

相关推荐
Billow_lamb3 小时前
MyBatis-Plus 的 条件构造器详解(超详细版)
java·mybatis
gugugu.5 小时前
Redis 渐进式 Rehash 深度剖析:如何实现平滑扩容与数据一致性
数据库·redis·缓存
A Mr Yang5 小时前
JAVA 对比老、新两个列表,找出新增、修改、删除的数据
java·开发语言·spring boot·后端·spring cloud·mybatis
武子康6 小时前
Java-190 EVCache入门:Netflix 级分布式缓存架构、性能指标与多区域部署全解析
java·redis·分布式·缓存·架构·guava·guava cache
2503_930123937 小时前
Redis群集的三种模式详解
数据库·redis·缓存
云和数据.ChenGuang7 小时前
openEuler 上安装与部署 Redis 的完整技术教程
数据库·redis·缓存
是萝卜干呀9 小时前
Redis
数据库·redis·缓存
妮妮喔妮9 小时前
redis热点key拆分和读多副本
数据库·redis·缓存
小雨下雨的雨9 小时前
第8篇:Redis缓存设计与缓存问题
java·redis·缓存
好学且牛逼的马11 小时前
【手写Mybatis | version0.0.3 附带源码 项目文档】
开发语言·php·mybatis