【MyBatis】验证多级缓存及 Cache Aside 模式的应用

文章目录

    • 前言
    • [1. 多级缓存的概念](#1. 多级缓存的概念)
      • [1.1 CPU 多级缓存](#1.1 CPU 多级缓存)
      • [1.2 MyBatis 多级缓存](#1.2 MyBatis 多级缓存)
    • [2. MyBatis 本地缓存](#2. MyBatis 本地缓存)
    • [3. MyBatis 全局缓存](#3. MyBatis 全局缓存)
      • [3.1 MyBatis 全局缓存过期算法](#3.1 MyBatis 全局缓存过期算法)
      • [3.2 CacheAside 模式](#3.2 CacheAside 模式)
    • 后记

前言

MyBatis 官方文档 中文版本没有翻译cache的部分,网上资料比较杂。

这里使用 Spock 测试框架验证下多级缓存,并探索 Mybatis 的 CacheAside 模式。注意:

  • 本文用 本地缓存 表示一级缓存,全局缓存 表示二级缓存
  • 用例仓库

1. 多级缓存的概念

多级缓存可以联系CPU的结构,离核心约近的一致性越高。

1.1 CPU 多级缓存

1.2 MyBatis 多级缓存

本地缓存默认开启,全局缓存需要使用 <cache/> 开启

By default, just local session caching is enabled that is used solely to cache data for the duration of a session. To enable a global second level of caching you simply need to add one line to your SQL Mapping file:

xml 复制代码
<mapper namespace="com.james.mapper.FileCacheMapper">
    <cache/> <!-- 声明该标签,全局缓存开启 -->
    <select id="select" resultType="java.lang.String">
        SELECT file_name FROM file
    </select>
</mapper>

2. MyBatis 本地缓存

用 Spring 注入的 mapper,调用一次select方法就会产生一个 sqlSession,没有利用到本地缓存。

groovy 复制代码
    def "未使用事务,第二次查询,不命中本地缓存"() {
        given:
        def list1 = mapper.select()
        def list2 = mapper.select()

        expect:
        list1 !== list2
    }

用事务包裹后,两次 select 共用一个 sqlSession,缓存命中

groovy 复制代码
 	def "使用事务,命中缓存"() {
        given:
        def list1 = []
        def list2 = []

        when:
        transaction.execute {
            list1 = mapper.select()
            list2 = mapper.select()
        }

        then:
        // 同一个事务使用同一个SqlSession,若引用相同则认为命中缓存
        list1 === list2
    }

note: groovy 中 list1 === list2 表示引用相同,list1 == list2 表示两个列表的内容相同

3. MyBatis 全局缓存

上文说到,没有事务保护的 select方法调用无法公用一个 sqlSession,所以利用不了本地缓存。

全局缓存的范围更大,只要是同一个mapper的调用,都会被缓存。

groovy 复制代码
    def "全局缓存默认关闭,需要在xml文件中使用 <cache/> 标签启用"() {
        given:
        def list1  = fileCacheMapper.select()
        def list2  = fileCacheMapper.select()

        expect:
        // 由于 SerializedCache.java:64 使用的是由byte[]序列化方式存储元素,所以实例的地址必然不同
        list1 !== list2
        list1 == list2
    }

3.1 MyBatis 全局缓存过期算法

值得关注的是 SOFT 和 WEAK 的类型,对应Java中软引用和弱引用。

软引用是在内存不足时GC可以回收,弱引用是下次GC即可回收(比软引用)积极。

3.2 CacheAside 模式

以下是 Mybatis 默认的全局缓存失效模式,也就是 Cache Aside 模式的应用。

  • 查询的时候,如果没有缓存,则写入。
  • 任何数据操作,使缓存失效。
xml 复制代码
<select ... flushCache="false" useCache="true"/>
<insert ... flushCache="true"/>
<update ... flushCache="true"/>
<delete ... flushCache="true"/>

后记

Cache Aside 并不能保证强一致性,不然也就不会有 Paxos 这种复杂的共识算法了。 ------ 《凤凰架构》

MyBatis 提供了缓存切口, 采用 Redis 会引入什么问题?

  • 多实例之间缓存重复的失效问题,查询时竞争写缓存的问题。
  • ORM框架与中间件耦合,违反单一职责。

万一遇到需强一致场景,如何增强?

  • 两个查询请求同时到来,此时缓存为空,需要将MySql数据写入缓存。此时会出现竞争写缓存的情况。用写锁来保证缓存内的数据跟数据库保持一致。
java 复制代码
public void query() {
	if (cache 命中) {
		retrun cache
	} 
	获取缓存写锁
	if (获取锁失败) {
		return 查数据库
	} 
	查数据库
	写缓存
	释放缓存写锁
}
相关推荐
fanTuanye1 小时前
redis 缓存穿透,缓存击穿,缓存雪崩
java·redis·缓存
Allen Bright4 小时前
【MyBatis-9】MyBatis分页插件PageHelper深度解析与实践指南
mybatis
在未来等你6 小时前
互联网大厂Java求职面试:电商商品推荐系统中的AI技术应用
java·缓存·kafka·推荐系统·向量数据库·jvm调优·spring ai
星星点点洲8 小时前
【Redis】谈谈Redis的设计
数据库·redis·缓存
柴薪之王、睥睨众生10 小时前
(自用)Java学习-5.8(总结,springboot)
java·开发语言·spring boot·学习·mybatis
Lion Long12 小时前
CodeBuddy 中国版 Cursor 实战:Redis+MySQL双引擎驱动〈王者荣耀〉战区排行榜
数据库·redis·mysql·缓存·腾讯云·codebuddy首席试玩官·codebuddy
唐僧洗头爱飘柔952721 小时前
【SSM-SSM整合】将Spring、SpringMVC、Mybatis三者进行整合;本文阐述了几个核心原理知识点,附带对应的源码以及描述解析
java·spring·mybatis·springmvc·动态代理·ioc容器·视图控制器
星星点点洲1 天前
【Redis】RedLock实现原理
redis·缓存
加什么瓦1 天前
Redis——数据结构
数据库·redis·缓存
lybugproducer1 天前
浅谈 Redis 数据类型
java·数据库·redis·后端·链表·缓存