Mybatis一级、二级缓存

在MyBatis中,缓存机制是减少数据库访问次数、优化查询性能的重要功能。MyBatis 提供了两级缓存:一级缓存和二级缓存。

一级缓存(Local Cache)

一级缓存是基于 SqlSession 级别的,即每个 SqlSession 对象都有自己的一级缓存。它默认是开启的,不需要任何配置。

原理解析

一级缓存的实现是通过保存查询结果到一个本地的Map中实现的,它的作用域是同一个 SqlSession。当在同一个 SqlSession 中执行相同的查询时,会从这个Map中直接获取到查询结果。

关键实现在 BaseExecutor 中:

java 复制代码
public abstract class BaseExecutor implements Executor {
  private final PerpetualCache localCache = new PerpetualCache("LocalCache");
  
  // ... 省略其他代码 ...
  
  public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey cacheKey, BoundSql boundSql) {
    if (localCache.getObject(cacheKey) != null) {
      // 从本地缓存获取结果
      return localCache.getObject(cacheKey);
    }
    // 查询数据库
    return doQuery(ms, parameter, rowBounds, resultHandler, boundSql);
  }
}

代码演示

java 复制代码
try (SqlSession session = sqlSessionFactory.openSession()) {
  BlogMapper mapper = session.getMapper(BlogMapper.class);
  // 第一次查询
  Blog blog1 = mapper.selectBlog(101);
  // 第二次相同的查询将会从一级缓存中获取数据
  Blog blog2 = mapper.selectBlog(101);
}

在以上代码中,第二次调用 selectBlog 方法时,并不会执行真正的SQL查询,而是直接从一级缓存中返回结果。

二级缓存(Global Cache)

二级缓存是基于 namespace 级别的,即映射的同一个mapper的不同实例之间共享这个缓存。

启用二级缓存

首先需要在 MyBatis 配置文件中启用二级缓存:

xml 复制代码
<settings>
  <!-- 开启全局缓存 -->
  <setting name="cacheEnabled" value="true"/>
</settings>

然后在需要使用二级缓存的 mapper xml 中配置 <cache/> 标签:

xml 复制代码
<mapper namespace="com.example.mapper.BlogMapper">
  <!-- 启用二级缓存 -->
  <cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
</mapper>

原理解析

二级缓存的核心是 Cache 接口及其实现,如 PerpetualCacheLruCache 等。

在 MyBatis 的 CachingExecutor 类中使用了装饰器模式包装了 Executor 对象,以实现二级缓存的功能:

java 复制代码
public class CachingExecutor implements Executor {
  private final Executor delegate;
  private final TransactionalCacheManager tcm = new TransactionalCacheManager();
  
  // ... 省略其他代码 ...
  
  public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) {
    Cache cache = ms.getCache();
    if (cache != null) {
      flushCacheIfRequired(ms);
      if (ms.isUseCache() && resultHandler == null) {
        ensureNoOutParams(ms, boundSql);
        @SuppressWarnings("unchecked")
        List<E> list = (List<E>) tcm.getObject(cache, key);
        if (list == null) {
          list = delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
          tcm.putObject(cache, key, list); // issue #578 and #116
        }
        return list;
      }
    }
    return delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
  }
}

代码演示

java 复制代码
try (SqlSession session1 = sqlSessionFactory.openSession()) {
  BlogMapper mapper = session1.getMapper(BlogMapper.class);
  Blog blog1 = mapper.selectBlog(101);
  session1.commit(); // 必须commit才能将结果放入二级缓存
}

try (SqlSession session2 = sqlSessionFactory.openSession()) {
  BlogMapper mapper = session2.getMapper(BlogMapper.class);
  Blog blog2 = mapper.selectBlog(101); // 这里将会使用二级缓存的数据
}

在以上代码中,虽然session1session2是两个不同的SqlSession实例,但是它们共享相同的mapper命名空间下的二级缓存。

  • 一级缓存仅在一个SqlSession内部有效,可以通过SqlSession.clearCache()清空。
  • 二级缓存跨SqlSession有效,但需要显示配置,且必须提交事务后才会将数据存入缓存。
  • 缓存中存储的是数据的副本,而不是直接引用,这就意味着缓存中的数据可能与数据库中的数据有所不同。
  • 缓存配置可设置淘汰策略(eviction)、清理间隔(flushInterval)、缓存大小(size)以及是否只读(readOnly)。
  • 二级缓存应谨慎使用,特别是在有多个事务都可能修改同一数据时,因为它可能导致脏读、不可重复读和幻读。
  • 在使用二级缓存时,缓存对象必须实现Serializable接口,因为它们可能需要从内存中序列化到磁盘。

理解和正确使用MyBatis的缓存机制,可以显著提升应用程序的性能,但同时也需要注意保证数据的一致性。

相关推荐
candyTong9 小时前
Claude Code 的 Edit 工具是怎么工作的
javascript·后端·架构
GetcharZp10 小时前
GitHub 2.4 万 Star!D2 正在重新定义程序员画图方式
后端
zhangxingchao12 小时前
多 Agent 架构到底怎么选?从 Claude Agent Teams、Cognition/Devin 到工程落地原则
前端·人工智能·后端
IT_陈寒12 小时前
SpringBoot那个自动配置的坑,害我排查到凌晨三点
前端·人工智能·后端
ServBay12 小时前
OpenCode 和它的7款必备插件
后端·github·ai编程
ping某12 小时前
逐字节拆解 tcpdump
后端
阿凡98073012 小时前
花 100 dollar,用 Claude 打通 EasyEDA&Fusion 双向同步
后端·程序员
irving同学4623812 小时前
从零搭建生产级 RAG:Embedding、Chunking、Hybrid Search 与 Reranker
前端·后端
她的男孩13 小时前
从零搭一个企业后台,为什么我把能力拆成 Starter 和 Plugin
java·后端·架构
胡志辉13 小时前
本地 AI 编码助手从 0 配起来:先选模型,再接 Ollama、VS Code、Claude Code 和 Codex
前端·后端