Mybatis 一级缓存和二级缓存原理区别


Posted in Java/Android onSeptember 23, 2022

Mybatis 一级缓存和二级缓存原理区别

Java面试经常问到Mybatis一级缓存和二级缓存,今天就给大家重点详解Mybatis一级缓存和二级缓存原理与区别

Mybatis缓存

缓存就是内存中的数据,常常来自对数据库查询结果的保存,使用缓存可以避免频繁与数据库进行交互,从而提高查询响应速度。

MyBatis 提供了对缓存的支持,分为一级缓存和二级缓存,如下图所示:

Mybatis 一级缓存和二级缓存原理区别

我们先大致了解下MyBatis一级缓存与MyBatis 二级缓存:

一级缓存:SqlSession级别的缓存,缓存的数据只在SqlSession内有效。

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

下面我们再分别详解两者的原理与区别。

Mybatis一级缓存

1.为什么需要Mybatis一级缓存

当我们使用Mybatis进行数据库的操作时候,会创建一个SqlSession来进行一次数据库的会话,会话结束则关闭SqlSession对象。

如果我们很有可能多次查询完全相同的sql语句,每一次查询都查询一次数据库,那查询数据库代价是比较大的,这会导致系统的资源浪费。

为了解决这个问题,Mybatis对每一次会话都添加了缓存操作,不用相同的SQL每次都需要查询数据库,这就是Mybatis一级缓存的作用。

2.Mybatis一级缓存的实现

我们知道对SqlSession的操作,mybatis内部都是通过Executor来执行的,Executor的生命周期和SqlSession是一致的。

Mybatis在Executor中创建了本地缓存(一级缓存),如下图所示:

Mybatis 一级缓存和二级缓存原理区别

大致的流程如下:

第一次查询用户id信息,先去缓存中查询是否有,如果没有,从数据库中查询用户信息,得到用户信息后在将用户信息储存到一级缓存中。

如果sqlSession去执行commit操作(插入、更新、删除),清空sqlSession中的一级缓存,保证缓存中始终保存的是最新的信息,避免脏读。

第二次查询用户id信息,先去缓存中查询,如缓存中有,直接从缓存中获取。

注意:两次查询须在同一个sqlsession中完成,否则将不会走mybatis的一级缓存。

在mybatis与spring进行整合开发时,事务控制在service中进行,重复调用两次servcie将不会走一级缓存,因为在第二次调用时session方法结束,SqlSession就关闭了。

3.Mybatis一级缓存配置

mybatis一级缓存的范围有SESSION和STATEMENT两种,默认是SESSION。

如果不想使用一级缓存,可以把一级缓存的范围指定为STATEMENT,这样每次执行完一个Mapper中的语句后都会将一级缓存清除。

如果需要更改一级缓存的范围,可以在Mybatis的配置文件中,在下通过localCacheScope指定。

<setting name="localCacheScope" value="STATEMENT"/>

Mybatis二级缓存

1.为什么需要Mybatis二级缓存?

MyBatis 一级缓存最大的共享范围就是一个SqlSession内部,那么如果多个 SqlSession 需要共享缓存,则需要开启二级缓存。

2.Mybatis二级缓存的实现

开启二级缓存后,会使用 CachingExecutor 装饰 Executor,进入一级缓存的查询流程前,先在 CachingExecutor 进行二级缓存的查询,具体的工作流程如下所示。

Mybatis 一级缓存和二级缓存原理区别

二级缓存开启后,同一个 namespace 下的所有操作语句,都影响着同一个 Cache,即二级缓存被多个 SqlSession 共享,是一个全局的变量。

当开启缓存后,数据的查询执行的流程就是 二级缓存 -> 一级缓存 -> 数据库。

MyBatis 是默认关闭二级缓存的,因为对于增删改操作频繁的话,那么二级缓存形同虚设,每次都会被清空缓存。

Mybatis一级缓存与二级缓存的区别

1)一级缓存 Mybatis的一级缓存是指SQLSession,一级缓存的作用域是SQlSession, Myabits默认开启一级缓存。

在同一个SqlSession中,执行相同的SQL查询时;第一次会去查询数据库,并写在缓存中,第二次会直接从缓存中取。 当执行SQL时候两次查询中间发生了增删改的操作,则SQLSession的缓存会被清空。

每次查询会先去缓存中找,如果找不到,再去数据库查询,然后把结果写到缓存中。 Mybatis的内部缓存使用一个HashMap,key为hashcode+statementId+sql语句。Value为查询出来的结果集映射成的java对象。 SqlSession执行insert、update、delete等操作commit后会清空该SQLSession缓存。

2) Mybatis二级缓存是默认不开启的,作用于一个Application,是Mapper级别的,多个SqlSession使用同一个Mapper的sql能够使用二级缓存。

到此这篇关于Mybatis 一级缓存和二级缓存原理区别 的文章就介绍到这了,更多相关Mybatis 缓存内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
详解java如何集成swagger组件
Jun 21 Java/Android
Java循环队列与非循环队列的区别总结
Jun 22 Java/Android
Java集成swagger文档组件
Jun 28 Java/Android
一篇文章带你学习Mybatis-Plus(新手入门)
Aug 02 Java/Android
spring cloud 配置中心native配置方式
Sep 25 Java/Android
SpringBoot中HttpSessionListener的简单使用方式
Mar 17 Java/Android
Android在Sqlite3中的应用及多线程使用数据库的建议
Apr 24 Java/Android
Jmerte 分布式压测及分布式压测配置
Apr 30 Java/Android
Java处理延时任务的常用几种解决方案
Jun 01 Java/Android
AndroidStudio图片压缩工具ImgCompressPlugin使用实例
Aug 05 Java/Android
Java使用HttpClient实现文件下载
Aug 14 Java/Android
Java结构型设计模式之组合模式详解
Sep 23 Java/Android
Java实现贪吃蛇游戏的示例代码
Sep 23 #Java/Android
Java获取字符串编码格式实现思路
Sep 23 #Java/Android
java获取一个文本文件的编码(格式)信息
Sep 23 #Java/Android
JDK8中String的intern()方法实例详细解读
Sep 23 #Java/Android
Spring boot实现上传文件到本地服务器
Aug 14 #Java/Android
Spring Boot实现文件上传下载
Aug 14 #Java/Android
Springboot集成kafka高级应用实战分享
You might like
php获取url字符串截取路径的文件名和扩展名的函数
2010/01/22 PHP
分享一段PHP制作的中文拼音首字母工具类
2014/12/11 PHP
PHP CURL post数据报错 failed creating formpost data
2016/10/16 PHP
PHP用正则匹配form表单中所有元素的类型和属性值实例代码
2017/02/28 PHP
PHP 枚举类型的管理与设计知识点总结
2020/02/13 PHP
jquery ajax提交表单数据的两种实现方法
2010/04/29 Javascript
JavaScript中的稀疏数组与密集数组[译]
2012/09/17 Javascript
浅析JS刷新框架中的其他页面 &amp;&amp; JS刷新窗口方法汇总
2013/07/08 Javascript
javaScript实现浮点数转十六进制字符
2013/10/29 Javascript
加随机数引入脚本不让浏览器读取缓存
2014/09/04 Javascript
JavaScript基础语法、dom操作树及document对象
2014/12/02 Javascript
JS封装的自动创建表格的实现代码
2016/06/15 Javascript
Angular.js中处理页面闪烁的方法详解
2017/03/09 Javascript
jQuery日期范围选择器附源码下载
2017/05/23 jQuery
微信小程序实现点击按钮修改文字大小功能【附demo源码下载】
2017/12/06 Javascript
Selenium执行JavaScript脚本的方法示例
2020/12/31 Javascript
[26:50]2018完美盛典DOTA2表演赛
2018/12/17 DOTA
python计算一个序列的平均值的方法
2015/07/11 Python
Python 搭建Web站点之Web服务器网关接口
2016/11/06 Python
Python读取和处理文件后缀为.sqlite的数据文件(实例讲解)
2017/06/27 Python
详解Django+uwsgi+Nginx上线最佳实战
2019/03/14 Python
如何使用Python进行OCR识别图片中的文字
2019/04/01 Python
pycharm的python_stubs问题
2020/04/08 Python
jupyter notebook 实现matplotlib图动态刷新
2020/04/22 Python
python使用requests库爬取拉勾网招聘信息的实现
2020/11/20 Python
python中@contextmanager实例用法
2021/02/07 Python
5分钟让你掌握css3阴影、倒影、渐变小技巧(小编推荐)
2016/08/15 HTML / CSS
巴西图书和电子产品购物网站:Saraiva
2017/06/07 全球购物
几个数据库方面的面试题
2016/07/01 面试题
为什么UNION ALL比UNION快
2016/03/17 面试题
一个J2EE项目团队的主要人员组成是什么
2012/06/04 面试题
给男朋友的道歉信
2014/01/12 职场文书
领导接待方案
2014/03/13 职场文书
交通志愿者活动总结
2014/06/27 职场文书
毕业设计论文评语
2014/12/31 职场文书
分享python函数常见关键字
2022/04/26 Python