深入源码剖析 Mybatis 缓存机制

news2024/11/20 2:28:43

深入源码剖析 Mybatis 缓存机制

Mybatis 为了减轻数据库压力,提高数据库性能。提供了两级缓存机制

  • 一级缓存

    sqlSession 级别缓存,缓存的数据只在 sqlSession 内有效,一级缓存默认为我们开起来,不需要我们手动操作,而且你也关不掉,但是我们可以手动清楚缓存。

  • 二级缓存

    mapper 级别缓存,同一个 namespace 公用一个缓存,所以对 sqlSession 是共享的,需要我们手动开启。

一级缓存

我们先来模拟一下一级缓存的代码

@Test
public void firstLevelCache(){
  // 第一次查询id为1的用户
  User user1 = userMapper.findUserById(1);
  User user2 = userMapper.findUserById(1);
  System.out.println(user1==user2);
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LYKbDw4Y-1673518190439)(images/Mybatis 缓存/image-20230112153744189.png)]

日志输出只查询一次库,而且 user1 == user2 由此可以判断一级缓存默认是开启的。

一级缓存的底层就是一个 Key-Value 形式的 Map,在执行查询操作时,去缓存中查找是否存在,如果存在就直接返回,如果不存在,就去数据库查询然后存一个到缓存里。

缓存底层的 key 是由 statementid、params、boundSql、rowBounds 组成

注意,做增删改操作并事务提交就会刷新一级缓存

除了增删改操作,我们也可以手动刷新缓存

sqlSession.clearCache();

解读源码

通过 sqlSesson.clearCache(); 往下探查源码,最终我们找到这一段代码

public class PerpetualCache implements Cache {
    private final String id;
    private Map<Object, Object> cache = new HashMap();

 		// 省略若干...
    public void clear() {
        this.cache.clear();
    }
  	// ....
}

所以也印证了, Mybatis 一级缓存的底层就是一个 PerpetualCache 类下的一个 HashMap

那他的工作流程是怎样的呢?

有Mybatis底层代码基础的人会清楚 Mybatis 的操作类都在 Executor 类中,我们可以在其中找到

CacheKey createCacheKey(MappedStatement var1, Object var2, RowBounds var3, BoundSql var4);

这个方法 返回的 CacheKey 就是一级缓存中的 Key

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vBl7hkRA-1673518190441)(images/Mybatis 缓存/image-20230112161046698.png)]

通过这段代码我们可以知道 这个 CacheKey 是由 statementid、params、boundSql、rowBounds 组成

  • statementid 可以理解为 nameSpace 的值(包路径.方法名)
  • params 参数
  • boundSql 是Myabtis底层解析的 sql 语句
  • rowBounds 分页条件

这样,我们的 CacheKey 就这样生成了

因为我们已经理清逻辑,在执行查询时候,就会先去查找判断 CacheKey

由此,我们可以找到 Executor.query() 方法

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Dai4YLd0-1673518190441)(images/Mybatis 缓存/image-20230112163632641.png)]

二级缓存

二级缓存的原理和一级缓存差不多,第一次查询会吧数据放入缓存中,然后第二次查询则会直接去缓存中取,但是一级缓存基于 sqlSession 的,而二级缓存基于 mapper 文件的 namespace ,也就是说,多个sqlSession 可以共享一个 mapper 中的二级缓存区域,并且如果两个 mapper 的namespace 相同,即使是两个 namespace,那么两个 mapper 中执行的 sql 查询到的数据也将在相同的二级缓存区存储

开启二级缓存:

和一级缓存不一样,二级缓存需要手动开启

可以在sqMapConfig.xml 加入如下代码

<settings>
	<setting name="cacheENabled" value="true"/>
</settings>

其次在 UserMapper.xml 中开启缓存

<cache/>
<!-- or -->
<cache eviction="FIFO" flushInterval="60000" readOnly="false" size="1024"/>
  • evication 缓存回收侧率 默认是 LRU
    • LRU 最近最少使用,移除最长时间不被使用的对象
    • FIFO 先进先出,按对象进入缓存顺序来移除它们
    • SOFT 软引用,移除基于垃圾回收器和软引用规则对象
    • WEAK 弱引用,更积极地移除基于垃圾收集器和弱引用规则的对象
  • flushInterval 缓存刷新间隔 单位毫秒
  • readOnly 是否只读
    • true 只读
    • false (读写,默认)
  • size 缓存存放多少元素
  • type 指定自定义缓存的全类名,也可以实现Cache接口 要使用二级缓存,对应的实体类必须实例化接口

如果是基于注解开发,就没有 UserMapper 配置文件,就可以在 IUserMapper 类上添加注解

@CacheNameSpace

测试二级缓存

@Test
public void SecondLevelCache(){
  SqlSession sqlSession1 = sqlSessionFactory.openSession();
  SqlSession sqlSession2 = sqlSessionFactory.openSession();
  SqlSession sqlSession3 = sqlSessionFactory.openSession();

  IUserMapper mapper1 = sqlSession1.getMapper(IUserMapper.class);
  IUserMapper mapper2 = sqlSession2.getMapper(IUserMapper.class);
  IUserMapper mapper3 = sqlSession3.getMapper(IUserMapper.class);

  User user1 = mapper1.findUserById(1);
  sqlSession1.close(); //清空一级缓存
  User user2 = mapper2.findUserById(1);

  System.out.println(user1==user2);
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vVdcJviE-1673518190441)(images/Mybatis 缓存/image-20230112170847887.png)]

但这里为什么是false 呢?难道不是走的缓存吗?

不是的,二级缓存和一级缓存不一样,二级缓存缓存的是数据,底层重新创建了个对象,把值给塞了进去。我们通过sql日志可以看到,只有一次查库记录。因为不是同一个对象,所以这里是false。

注意,开启二级缓存相关的实体类是需要序列化的

开启二级缓存之后呢,数据存储介质多种多样,不一定在内存中,可能是磁盘。如果我们要再取这个缓存,就需要反序列化了。所以 mybatis 中的 实体都要去实现 Seriablizable 接口

useCache和flushCache

mybatis 中还可以设置 useCache 和 flushCache 选项

  • useCache 设置是否禁用二级缓存,在 statement 中设置 useCache = false 可以禁用当前 select 语句的二级缓存,每次查询都会去查库, 默认为 true
<select id="queryAll" useCache="false" resultType="user">
  select * from user 
</select>

这种情况针对每次查询完都需要新的数据,禁止二级缓存

在 mapper 的同一个 namespace 中,如果有其他 insert、update、delete 操作后需要刷新缓存,如果不执行则会出现脏读

设置 statement 参数 flushCache = true 属性,默认情况为 true,即刷新缓存,如果改为 false 则不会刷新,使用缓存时如果手动修改数据库中的查询会出现脏读

<select id="queryAll" flushCache="true" useCache="false" resultType="user">
	select * from user
</select>

一般下执行完 commit 操作都需要刷新缓存,flushCache =ture 表示刷新,这样可以避免数据库脏读,所以我们不用设置,默认即可

解读源码

二级缓存是通过 org.apache.ibatis.cache.impl.PerpetualCache 来实现的,其实就是一级缓存的类,但实际上是通过 PerpetualCache 实现的 Cache 接口来实现的 ,其实底层也是 HashMap

其实在 开启二级缓存的注解上可以写具体的 二级缓存实现类

@CacheNamespace(implementation = PerpetualCache.class)//开启二级缓存

这样只是更具体一些,当然,你也可以自定义二级缓存实现类

使用默认的 PerpetualCache 实现类和不写 implementation 参数效果是一样的

默认的 Mybatis 二级缓存是由缺陷的

在分布式环境下,服务器之间的缓存不能共用

我们可以通过分布式缓存方案来解决这个问题,redis、memcached、ehcache 等

Mybatis与Redis整合

mybatis 自己提供了一个 cache 接口,如果要实现自己的逻辑,实现 cache 接口开发就可以了,默认用 mybatis 提供的 cache 接口 redis 实现类就行,该类在 mybatis-redis 包中

<dependency>
  <groupId>org.mybatis.caches</groupId>
  <artifactId>mybatis-redis</artifactId>
  <version>1.0.0-beta2</version>
</dependency>

配置文件 redis.properties

redis.host=localhost
redis.port=6379
redis.connectionTimeout=5000
redis.password=
redis.database=1

修改二级缓存的引用类

import org.mybatis.caches.redis.RedisCache;
@CacheNamespace(implementation = RedisCache.class)//开启二级缓存

我们看一下 RedisCache 里面做了什么

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bUMNvWsm-1673518190442)(images/Mybatis 缓存/image-20230112175355791.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-iXcYkJ0m-1673518190442)(images/Mybatis 缓存/image-20230112180036614.png)]

这一块就是 RedisCache 对缓存数据存值和取值

由此我们可以知道 RedisCache 底层的数据类型也是 map 类型

关注公众号「Xiang想」

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/159457.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Dash搭建可视化网站】项目14:美国机场交通数据可视化看板制作步骤详解

美国机场交通数据可视化看板制作步骤详解1 项目效果图2 项目架构3 文件介绍和功能完善3.1 assets文件夹介绍3.2 app.py和index.py文件完善3.3 sider.py文件完善3.4 mapchart.py文件完善3.5 barchart.py文件完善3.6 api.py和api.ipynb文件完善4 样式修改4.1 整体样式修改4.2 sid…

联合证券|A股汽车板块爆发,北向资金半日净买入43亿

今日上午&#xff0c;A股商场震动胶着&#xff0c;轿车、电力设备、通信、机械设备等板块领涨。房地产、美容护理、商贸零售等板块领跌。 北向资金再度成为商场重要亮点之一&#xff0c;半个交易日净买入约43亿元。至此&#xff0c;2023年1月以来&#xff0c;北向资金累计净买入…

Jmeter常用函数

1、生成随机数函数 ${__Random(m,n)}&#xff0c;其中m,n参数是必填 2、 生成随机日期函数${__RandomDate(dateTimeFormat,from,end,locale,var)}&#xff0c;其中end是必选的&#xff0c;代表最大的日期&#xff1b; 3、 随机生成字符串函数${__RandomString(length,chars,)}…

Linux---vim编辑器

目录 1. vim的基本概念 2. vim正常/命令模式命令集 3. vim底行模式命令集 1. vim的基本概念 vim是Linux下一款常用编辑器&#xff0c;vim的三种模式(其实有好多模式&#xff0c;主要掌握这3种即可),分别是命令模式&#xff08;command mode&#xff09;、插入模式&#xff0…

Synchronized锁原理及 ConcurrentHashMap

文章目录一、Synchronized原理加锁过程锁消除锁粗化二、线程安全的集合类多线程环境使用ArrayList多线程环境使用队列多线程环境下使用哈希表一、Synchronized原理 我们表面看到的&#xff0c;两个线程针对同一对象加锁&#xff0c;就会产生阻塞等待&#xff0c;但实际我们的S…

2023我的创作纪念日

文章目录机缘收获日常憧憬机缘 这个博客还是我上大一的时候注册的&#xff0c;在大一、大二、大三期间更多的是为了方便搜索&#xff0c;学校里边的习题大部分是可以在CSDN上找到的。真正写博客是在大三下学习实习&#xff0c;当时为了方便记录实习中遇到的问题。在C站对我影响…

【Git 从入门到精通】2023最新版的Git安装与卸载每一步附详细讲解

文章目录安装1.下载Git2.开始安装卸载1.找到电脑中的Git2.卸载3.删除环境变量安装 1.下载Git 首先去官网下载Git安装包&#xff0c;可以直接在百度搜索Git&#xff0c;以下几个网站都可以。也可以点击直达&#xff0c;官网上下载如果不科学上网的话还是很慢的&#xff0c;所以…

[Zombodb那些事]Zombodb与ElasticSearch的Bulk通信

Zombodb与ElasticSearch的Bulk通信0.前言Zombodb是一个PostgreSQL插件&#xff0c;使用rust编写&#xff0c;支持pg14以下版本。Zombodb可以允许PostgreSQL查询ElasticSearch中的内容。本篇为《Zombodb那些事》第一篇&#xff0c;后面将更新其他部分内容。Zombodb会在pg数据库上…

智能文字识别技术推动彝文识别弘扬中华文化

前言 谈起图像识别自己颇有感触&#xff0c;因为之前的两段工作经历都和图像识别密切相关&#xff1b;之前一家公司的主营业务就是将历史上珍贵文献进行数字化&#xff1b;上家公司自己负责图像识别模块相关的工作&#xff1b;不但使用了第三方平台产品而且进行了自建&#xff…

设计模式相关内容介绍—UML

统一建模语言(Unified ModelingLanguage&#xff0c;UML)是用来设计软件的可视化建模语言。它的特点是简单、统一、图形化、能表达软件设计中的动态与静态信息。 UML从目标系统的不同角度出发&#xff0c;定义了用例图、类图、对象图、状态图、活动图、时序图、协作图、构件图、…

经过2022年这大环境,我学会了如何管理我的领导

2022年这大环境&#xff0c;可以说是我干软件开发这些年来&#xff0c;经历的最残酷的一年&#xff0c;所以做为职场软件开发一员的我&#xff0c;不得不修炼一下真本事。 很多时候不是你不努力&#xff0c;不是你连mysql连的不溜&#xff0c;不是你布局页面布局的不精细&#…

16.Isaac教程--Codelets详解

Codelets详解 ISAAC教程合集地址: https://blog.csdn.net/kunhe0512/category_12163211.html 文章目录Codelets详解Codelets 和 tick接收消息传输消息方便的 ToProto/FromProto 函数配置参数应用程序 JSON子图姿态组件是机器人应用程序的基本构建块。 Isaac SDK 包含可在您的应…

「数据结构详解·九」图的初步

「数据结构详解一」树的初步「数据结构详解二」二叉树的初步「数据结构详解三」栈「数据结构详解四」队列「数据结构详解五」链表「数据结构详解六」哈希表「数据结构详解七」并查集的初步「数据结构详解八」带权并查集 & 扩展域并查集「数据结构详解九」图的初步 注意&…

基于JavaWEB SSM SpringBoot婚纱影楼摄影预约网站设计和实现

基于JavaWEB SSM SpringBoot婚纱影楼摄影预约网站设计和实现 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末…

天宝营养冲刺深交所IPO:业绩明显波动,深创投是股东

撰稿|汤汤 来源|贝多财经 近日&#xff0c;贝多财经发现&#xff0c;天宝动物营养科技股份有限公司&#xff08;下称“天宝营养”&#xff09;递交预披露更新招股书&#xff0c;准备在深圳证券交易所主板上市&#xff0c;红塔证券为其独家保荐人。本次冲刺上市&#xff0c;天…

《Protein Actions Principles and Modeling》-《蛋白质作用原理和建模》中文分享(16)

​《Protein Actions Principles and Modeling》-《蛋白质作用原理和建模》 本人能力有限&#xff0c;如果错误欢迎批评指正。 第四章&#xff1a;Protein Binding Leads to Biological Actions &#xff08;蛋白质的结合会产生生物作用&#xff09; -在变构中&#xff0c;…

大神推荐,这几个电脑实用技巧,让你电脑用起来更加流畅舒服

电脑在我们的日常生活中&#xff0c;往往承担着“办公学习”的作用&#xff01;所以我们应该掌握哪些常用、好用的电脑使用技巧呢&#xff1f;今天就给大家分享下&#xff0c;我日常在使用电脑过程中&#xff0c;经常会使用到的几个电脑使用技巧&#xff01;第一&#xff1a;快…

基于FPGA的UDP 通信(三)

目录 引言 设计框图 UDP接收模块 设计源码 TEST BENCH 仿真结果 引言 前文链接&#xff1a; 基于FPGA的UDP 通信&#xff08;一&#xff09; 基于FPGA的UDP 通信&#xff08;二&#xff09; 本文基于FPGA设计千兆以太网通信模块&#xff1a;FPGA接收上位机数据。后续…

端到端的传输协议

&#xff08;一&#xff09;如何在一条物理链路上进行有效和可靠的数据传输 ——数据链路层传输协议 &#xff08;1&#xff09;标识高层送下来的数据块的起止、特定内容&#xff08;例如校验比特&#xff09;的位置 ——组帧技术 &#xff08;2&#xff09;如何发现传输中的错…

数据结构---线性表

刘佳瑜*&#xff0c;王越 *, 黄扬* , 张钊* (淮北师范大学计算机科学与技术学院&#xff0c;安徽 淮北) *These authors contributed to the work equllly and should be regarded as co-first authors. &#x1f31e;欢迎来到数据结构的世界 &#x1f308;博客主页&#xff1…