news 2026/4/28 5:07:41

Mybatis的一级缓存和二级缓存,看完不再懵逼

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Mybatis的一级缓存和二级缓存,看完不再懵逼

缓存是什么?

缓存其实就是存储在内存中的临时数据,这里的数据量会比较小,一般来说,服务器的内存也是有限的,不可能将所有的数据都放到服务器的内存里面,所以, 只会把关键数据放到缓存中,缓存因为速度快,使用方便而出名!

为什么需要缓存

BS架构里面,用户的所有操作都是对数据库的增删改查,其中查询的操作是最多的,但如果用户想要某个数据时每次都去数据库查询,这无疑会增加数据库的压力,而且获取时间效率也会降低,所以为了解决这些问题,缓存应用而生,使用了缓存之后,服务器只需要查询一次数据库,然后将数据保存到服务器主机的内存中,以后读取时就直接取内存中的数据,而不需要每次都查数据库,这种方案除了降低数据库压力之外,还提高了响应速度,简直一箭双雕哇~

哪些数据会放到缓存

通常情况下,都会将那些变化较少且经常用到的数据会放到缓存中,比如像字典、系统参数、有固定值的状态码等等;另外将用户保存到缓存也是一种很好的策略,这样登录的时候就可以极速响应了;

mybatis一级缓存

mybatis 的缓存分为2类,分别是一级缓存和二级缓存,一级缓存是默认开启的,它在一个sqlSession会话里面的所有查询操作都会保存到缓存中,一般来说一个请求中的所有增删改查操作都是在同一个sqlSession里面的,所以我们可以认为每个请求都有自己的一级缓存,如果同一个sqlSession会话中2 个查询中间有一个 insert 、update或delete 语句,那么之前查询的所有缓存都会清空;

Reader reader = Resources.getResourceAsReader("config/configuration.xml"); //创建数据工厂 SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory sqlSessionFactory = builder.build(reader); SqlSession sqlSession = sqlSessionFactory.openSession(true); // 。。。。。。 // 这中间所走的所有查询操作都会进行缓存,一旦关闭sqlSession会话,缓存则会刷新 //释放会话 sqlSession.clearCache(); // 关闭会话 sqlSession.close();

一级缓存流程图

[图片上传失败...(image-1956eb-1649686882541)]

我们来试验一把

在mybatis 的配置文件中加入以下配置,开启sql日志,每一个sql代表请求了一次数据库,这样我们就可以根据sql来判断是否使用了缓存

<settings> <!--标准的日志工厂实现类,打印sql日志--> <setting name="logImpl" value="STDOUT_LOGGING"/> </settings>

示例代码

public static void main(String[] args) throws IOException { // 加载mybatis配置文件 Reader reader = Resources.getResourceAsReader("config/configuration.xml"); //创建数据工厂 SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory sqlSessionFactory = builder.build(reader); SqlSession sqlSession = sqlSessionFactory.openSession(true); // 获取mapper接口对象 UserMapper mapper = sqlSession.getMapper(UserMapper.class); // 查询第一次 User user = mapper.selectByPrimaryKey("3rfrf34r34"); // 第二次查询 User user1 = mapper.selectByPrimaryKey("3rfrf34r34"); System.out.println("两个user对象是否相等:"+(user == user1)); //释放会话 sqlSession.clearCache(); sqlSession.close(); }

打印结果

image.png

根据结果可以看到,代码中执行了2次查询, 但实际运行时只查询了一次数据库,第二次获取数据时直接从缓存中读取,并且2次读取的数据都是一样的,到这里,一级缓存就已经生效了;

接下来我们来测试第二种情况 :查询 -> 修改 -> 查询

实例代码

public static void main(String[] args) throws IOException { // 加载mybatis配置文件 Reader reader = Resources.getResourceAsReader("config/configuration.xml"); //创建数据工厂 SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory sqlSessionFactory = builder.build(reader); SqlSession sqlSession = sqlSessionFactory.openSession(true); // 获取mapper接口对象 UserMapper mapper = sqlSession.getMapper(UserMapper.class); // 查询第一次 User user = mapper.selectByPrimaryKey("3rfrf34r34"); // 修改 mapper.updateByPrimaryKey(user); // 第二次查询 User user1 = mapper.selectByPrimaryKey("3rfrf34r34"); System.out.println("两个user对象是否相等:"+(user == user1)); //释放会话 sqlSession.clearCache(); sqlSession.close(); }

打印结果

image.png

控制台打印了三次sql,其中第一个查询和第三个查询是一样的,但是并没有使用缓存,为什么会这样呢?因为每次增删改操作都有可能会改变原来的数据,所以必须刷新缓存;

二级缓存

二级缓存是全局的,也就是说;多个请求可以共用一个缓存,二级缓存需要手动开启,有2种方式配置二级缓存,

  • 缓存会先放在一级缓存中,当sqlSession会话提交或者关闭时才会将一级缓存刷新到二级缓存中;

  • 开启二级缓存后,用户查询时,会先去二级缓存中找,找不到在去一级缓存中找;

二级缓存流程图

image.png

第一种配置方式

单个mapper配置,主需要在需要开启二级缓存的mapper.xml文件中加入以下配置即可开启

<!-- 开启单个mapper的二级缓存,也叫全局缓存--> <cache />

注意一定要加到xxMapper.xml的文件内,千万不要加到mybatis 的主配置文件里面了,会报错的

image.png

第二种配置方式

所有的mapper都开启二级缓存,在mybatis.xml中加入以下配置即可

<settings> <!-- 开启所有mapper的二级缓存 --> <!--<setting name="cacheEnabled" value="true" />--> </settings>

image.png

示例代码

public static void main(String[] args) throws IOException { // 加载mybatis配置文件 Reader reader = Resources.getResourceAsReader("config/configuration.xml"); //创建数据工厂 SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder(); SqlSessionFactory sqlSessionFactory = builder.build(reader); // 第一个会话 SqlSession sqlSession = sqlSessionFactory.openSession(true); // 获取会话一的mapper接口对象 UserMapper mapper = sqlSession.getMapper(UserMapper.class); // 第一次查询 User user = mapper.selectByPrimaryKey("3rfrf34r34"); //释放第一个会话 sqlSession.clearCache(); sqlSession.close(); // 第二个会话 SqlSession sqlSession2 = sqlSessionFactory.openSession(true); // 获取会话二的mapper接口对象 UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class); // 第二次查询 User user1 = mapper2.selectByPrimaryKey("3rfrf34r34"); // 释放第二个会话 sqlSession2.clearCache(); sqlSession2.close(); }

打印结果

image.png

打印结果很明显,2次查询,但是日志显示只查询了一次数据库, 第二次是从缓存中获取的数据,至此,二级缓存已开启!

注意事项

  • 另外,缓存还有以下几种情况需要注意
  • 映射语句文件中的所有 select 语句的结果将会被缓存。
  • 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。
  • 缓存会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。
  • 缓存不会定时进行刷新(也就是说,没有刷新间隔)。
  • 缓存会保存列表或对象(无论查询方法返回哪种)的 1024 个引用。
  • 缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。

AI大模型学习福利

作为一名热心肠的互联网老兵,我决定把宝贵的AI知识分享给大家。 至于能学习到多少就看你的学习毅力和能力了 。我已将重要的AI大模型资料包括AI大模型入门学习思维导图、精品AI大模型学习书籍手册、视频教程、实战学习等录播视频免费分享出来。

一、全套AGI大模型学习路线

AI大模型时代的学习之旅:从基础到前沿,掌握人工智能的核心技能!

因篇幅有限,仅展示部分资料,需要点击文章最下方名片即可前往获取

二、640套AI大模型报告合集

这套包含640份报告的合集,涵盖了AI大模型的理论研究、技术实现、行业应用等多个方面。无论您是科研人员、工程师,还是对AI大模型感兴趣的爱好者,这套报告合集都将为您提供宝贵的信息和启示。

因篇幅有限,仅展示部分资料,需要点击文章最下方名片即可前往获

三、AI大模型经典PDF籍

随着人工智能技术的飞速发展,AI大模型已经成为了当今科技领域的一大热点。这些大型预训练模型,如GPT-3、BERT、XLNet等,以其强大的语言理解和生成能力,正在改变我们对人工智能的认识。 那以下这些PDF籍就是非常不错的学习资源。


因篇幅有限,仅展示部分资料,需要点击文章最下方名片即可前往获

四、AI大模型商业化落地方案

因篇幅有限,仅展示部分资料,需要点击文章最下方名片即可前往获

作为普通人,入局大模型时代需要持续学习和实践,不断提高自己的技能和认知水平,同时也需要有责任感和伦理意识,为人工智能的健康发展贡献力量

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/28 4:42:42

新手入门必看:BAAI/bge-m3 WebUI界面操作与调试实战指南

新手入门必看&#xff1a;BAAI/bge-m3 WebUI界面操作与调试实战指南 1. 引言 1.1 学习目标 本文旨在为初学者提供一份完整的 BAAI/bge-m3 模型 WebUI 操作与调试实战指南。通过本教程&#xff0c;您将掌握如何使用集成 WebUI 的 bge-m3 镜像进行语义相似度分析&#xff0c;理…

作者头像 李华
网站建设 2026/4/23 14:02:58

MGeo地址相似度识别性能报告:长尾地址匹配能力评估

MGeo地址相似度识别性能报告&#xff1a;长尾地址匹配能力评估 1. 技术背景与评估目标 在地理信息处理、位置服务和数据融合等应用场景中&#xff0c;地址相似度识别是实现实体对齐的核心技术之一。由于中文地址存在表述多样、结构不规范、别名广泛等特点&#xff0c;尤其是“…

作者头像 李华
网站建设 2026/4/18 10:38:25

轻松搞定长文本标准化|基于FST ITN-ZH镜像的高效转换方案

轻松搞定长文本标准化&#xff5c;基于FST ITN-ZH镜像的高效转换方案 在自然语言处理的实际应用中&#xff0c;中文逆文本标准化&#xff08;Inverse Text Normalization, ITN&#xff09;是一个常被忽视但至关重要的环节。语音识别系统输出的往往是口语化、非结构化的表达&am…

作者头像 李华
网站建设 2026/4/24 22:34:17

电力系统三相短路故障分析:Simulink仿真与报告

电力系统三相短路故障分析simulink仿真加报告打开Simulink随便拖几个模块就能搭三相短路模型&#xff1f;别急&#xff0c;先看这个案例里藏着哪些电力人必懂的玄机。咱们直接从搭建三相短路故障模型开始&#xff0c;先拖出Power System Blockset里的三相电压源模块。设置电源参…

作者头像 李华
网站建设 2026/4/26 16:36:30

YOLOv9训练总失败?镜像免配置部署案例一文解决

YOLOv9训练总失败&#xff1f;镜像免配置部署案例一文解决 在深度学习目标检测领域&#xff0c;YOLOv9 一经发布便因其出色的性能和创新的可编程梯度信息&#xff08;PGI&#xff09;机制受到广泛关注。然而&#xff0c;许多开发者在尝试从源码部署 YOLOv9 时&#xff0c;常常…

作者头像 李华
网站建设 2026/4/26 21:11:26

实测NewBie-image-Exp0.1:3.5B参数模型带来的动漫创作革命

实测NewBie-image-Exp0.1&#xff1a;3.5B参数模型带来的动漫创作革命 1. 引言&#xff1a;从复杂部署到开箱即用的生成体验 在当前AI图像生成领域&#xff0c;尤其是面向高质量动漫内容创作方向&#xff0c;开发者和研究人员常常面临一个共同挑战&#xff1a;环境配置复杂、…

作者头像 李华