面试官:说一下 MyBatis 的一级缓存和二级缓存 ?

目录

1. MyBatis 的缓存机制

2. 为什么不默认开启 MyBatis 的二级缓存

3. MyBatis 如何开启二级缓存


1. MyBatis 的缓存机制

MyBayis 中包含两级缓存:一级缓存和二级缓存

1. 一级缓存是 SqlSession 级别的,是 MyBatis 自带的缓存功能,默认是开启的,并且无法关闭,所以当有两个 SqlSession 执行相同的 SQL 时,就没有用到一级缓存,而是查询了两次数据库。

2. 二级缓存是 Mapper 级别的,只要是同一个 Mapper,无论使用多少个 SqlSession 进行操作,数据都是共享的,所以多个 SqlSession 可以共享二级缓存。但是 MyBatis 的二级缓存默认是关闭的,需要时可以手动开启。此外,二级缓存还可以使用第三方的缓存,例如:Ehcache。

2. 为什么不默认开启 MyBatis 的二级缓存

为什么不默认开启二级缓存呢 ? 缓存不是可以加速程序的查询性能吗 ?

MyBatis 不默认开启二级缓存的原因有以下几点:

1. 缓存粒度过大:因为二级缓存是一个全局缓存,可以缓存多个不同的查询结果集。默认情况下,MyBatis 是不知道哪些查询结果需要缓存,哪些查询结果不需要缓存。当开启二级缓存后,所有的查询结果都尝试使用缓存,这就可能会导致缓存的数据不准确或者不一致性。

例如上图,三次查询操作查询到的结果可能不一致,此时 MyBatis 默认不知道缓存哪个查询结果,这样就存在缓存不准确的风险。

2. 并发性问题:在多线程情况下,开启二级缓存,如果没有及时清空或刷新缓存,就可能会导致缓存和数据库数据不一致性问题。

此处的并发性问题可以类比到 Redis 和 MySQL 数据不一致性问题 :

关于多线程情况下的缓存和数据库不一致性问题以及解决方案,可以看我的这篇博客:https://blog.csdn.net/xaiobit_hl/article/details/132453064

3. 内存占用问题:开启二级缓存之后,缓存的数据需要占用大量的内存空间,如果没有合适的策略来管理缓存,可能就会导致内存占用过多的问题。

4. 复杂性问题:二级缓存的配置需要考虑诸多因素,例如:缓存的刷新以及缓存的清理,这都需要较好的缓存策略来处理,这就加大了开发的复杂性,并且有可能引入新的问题。

        基于以上问题,MyBatis 选择默认关闭二级缓存,当开发人员确认某些查询可以受益于缓存时,再手动开启二级缓存来使用即可(把主动权交给了开发人员)。

3. MyBatis 如何开启二级缓存

MyBatis 中开启二级缓存需要两步操作:

  1. 在 mapper 对应的 xml 中添加 <cache> 标签;
  2. 在 xml 中给需要缓存的标签设置 useCache="true"。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.demo.mapper.UserMapper"><cache/><select id="getAll" resultType="UserInfo" useCache="true">select count(*) from userinfo;</select>
</mapper>

进行单元测试:

@SpringBootTest
class UserMapperTest {@Resourceprivate UserMapper userMapper;@Testvoid getAll() {int ret1 = userMapper.getAll();System.out.println("查询结果:" + ret1);int ret2 = userMapper.getAll();System.out.println("查询结果:" + ret2);}
}

【说明】

        此处在外部方法,调用两次 getAll()  方法,它一定会使用两个 SqlSession,只有在一个 getAll() 方法里面执行两条相同的 SQL 时,才会使用同一个 SqlSession。

如何判断是否走缓存?(properties 文件中配置执行打印 SQL)

  • 如果两次查询都打印了 SQL 语句,说明没有走缓存,
  • 如果第一次查询打印了 SQL 语句,第二次没有打印,说明第二次查询走的是缓存。

【执行结果】

【结果分析】

        执行结果很显然,第二次查询走了缓存。虽然调用了两次 getAll() 方法,使用了两个 SqlSession,但是因为我前面开启了二级缓存,二级缓存的作用域是整个 mapper,所以不管是用了几个 SqlSession,第二次查询肯定会走缓存。

        如果关闭二级缓存那么两次查询都会查询数据库(创建了两个 SqlSession,都打印了 SQL),执行结果如下:

【一级缓存示例】

        在关闭二级缓存的情况下,还想使得第二次查询走缓存,可以通过在方法上加一个 @Transactional 注解就可以做到。

        当我们在方法上加上 @Transactional 注解的时候,事物里面的所有方法就会使用一个 SqlSession 来进行操作,那么第二次查询也就自然而然会走缓存了。

关于 MyBatis  缓存的清除策略,明天再继续补充....

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在CentOS7上使用Docker安装和部署RabbitMQ

&#x1f680; 1 拉取RabbitMQ Docker镜像 首先&#xff0c;使用Docker命令从Docker Hub拉取RabbitMQ官方镜像。打开终端并运行以下命令&#xff1a; docker pull rabbitmq&#x1f680; 2 创建RabbitMQ容器 一旦镜像下载完成&#xff0c;使用以下命令创建RabbitMQ容器&…

x86 汇编手册快速入门

本文翻译自&#xff1a;Guide to x86 Assembly 在阅读 Linux 源码之前&#xff0c;我们需要有一些 x86 汇编知识。本指南描述了 32 位 x86 汇编语言编程的基础知识&#xff0c;包括寄存器结构&#xff0c;数据表示&#xff0c;基本的操作指令&#xff08;包括数据传送指令、逻…

【leetcode 力扣刷题】数学题之数的开根号:二分查找

用二分查找牛顿迭代解决开根号 69. x的平方根367. 有效的完全平方数 69. x的平方根 题目链接&#xff1a;69. x的平方根 题目内容&#xff1a; 题意是要我们求一个数的算数平方根&#xff0c;但是不能使用内置函数&#xff0c;那么我们就暴力枚举。我们知道如果y>2的话&am…

PO设计模式是selenium自动化测试中最佳的设计模式之一

Page Object Model&#xff1a;PO设计模式是selenium自动化测试中最佳的设计模式之一&#xff0c;主要体现在对界面交互细节的封装&#xff0c;也就是在实际测试中只关注业务流程就OK了传统的设计中&#xff0c;在新增测试用例之后&#xff0c;代码会有以下几个问题&#xff1a…

经管博士科研基础【16】一元二次函数的解的公式

1. 一元二次函数的形式 2. 一元二次函数的图形与性质 一元二次函数的图像是一条抛物线&#xff0c;图像定点公式为(-b/2a,4ac-b*b/4a)&#xff0c;对称轴位直线x-b/2a。 3. 求根公式 形如ax*xb*xc0的一元二次方程&#xff0c;其求根公式为&#xff1a; 4. 韦达定理 如果x1和…

ScreenToGif-动图制作软件实用操作

ScreenToGif官网&#xff1a;ScreenToGif ⭕第一步&#xff1a;启动页面 ⭕第二步&#xff1a;选项 &#x1f95d;录像机-捕获频率选择手动-播放延迟1000ms(可以任意) ⭕第三步&#xff1a;录像机开始录屏 &#x1f95d;我们调整录屏的大小后&#xff0c;打开画图&#xff0c…

剪枝基础与实战(5): 剪枝代码详解

对模型进行剪枝,我们只对有参数的层进行剪枝,我们基于BatchNorm2d对通道重要度 γ \gamma γ参数进行稀释训练。对BatchNorm2d及它的前后层也需要进行剪枝。主要针对有参数的层:Conv2d、BatchNorm2d、Linear。但是我们不会对Pool2d 层进行剪枝,因为Pool2d只用来做下采样,没…

【进阶篇】MySQL分库分表详解

文章目录 0. 前言1. 垂直分库分表2. 水平分库分表 1. 理解过程及实现方案问题讨论衍生出分库分表策略借助成熟组件使用分库分表阶段完成后面临的问题1. 异地多活问题2. 数据迁移问题3. 分布式事务问题4. join查询的问题 分库分表的策略实现示例 2. 参考文档 0. 前言 假设有一个…

46、SpringBoot输入校验--JSR 303

★ Spring Boot的输入校验 springboot支持两种校验方式&#xff1a;1. Spring原生提供的 Validation&#xff0c;这种验证方式需要开发者手写验证代码&#xff0c;比较繁琐。就是普通的if判断2. 使用JSR 303的校验&#xff0c;这种验证方式只需使用注解、即可以声明式的方式进…

4G版本云音响设置教程阿里云平台版本

4G版本云音响设置教程介绍 第一章 介绍了在阿里云物联网平台生一个设备使用的三元素 第二章 转换阿里云三元素 为MQTT参数&#xff0c;并下载到设备中 第三章 阿里云物联网套件协议使用说明&#xff0c;如何发送数据至设备并播放 本文目录引导 目录 4G版本云音响设置教程介…

利用frps搭建本地自签名https服务的透传

nginx的搭建就不介绍了&#xff0c;教程很多&#xff0c;基本上油手就会。 在本例中&#xff0c;frp服务器的域名是 www.yourfrp.com&#xff0c;同时也是反向代理nginx服务器; 本地网站要用的域名&#xff1a; test.abcd.com 请事先将 test.abcd.com 解析到 frp所在服务器…

大数据面试题:MapReduce压缩方式

面试题来源&#xff1a; 《大数据面试题 V4.0》 大数据面试题V3.0&#xff0c;523道题&#xff0c;679页&#xff0c;46w字 可回答&#xff1a;1&#xff09;Hadoop常见的压缩算法有哪些&#xff1f; 问过的一些公司&#xff1a;网易云音乐(2022.11)&#xff0c;阿里(2020.…

重写 UGUI

重写Button using UnityEngine; using UnityEngine.UI; public class MyButton : Button {[SerializeField] private int _newNumber; }using UnityEditor;//编辑器类在UnityEditor命名空间下。所以当使用C#脚本时&#xff0c;你需要在脚本前面加上 "using UnityEditor&q…

华为云云服务器评测 [Vue3 博物馆管理系统] 使用Vue3、Element-plus菜单组件构建轮播图

系列文章目录 第一章 定制上中下&#xff08;顶部菜单、底部区域、中间主区域显示&#xff09;三层结构首页 第二章 使用Vue3、Element-plus菜单组件构建菜单 第三章 使用Vue3、Element-plus菜单组件构建轮播图 [第四章 使用Vue3、Element-plus菜单组件构建组图文章] 华为云云…

如何中mac上安装多版本python并配置PATH

摘要 mac 默认安装的python是 python3&#xff0c;但是如果我们需要其他python版本时&#xff0c;该怎么办呢&#xff1f; 例如&#xff1a;需要python2 版本&#xff0c;如果使用homebrew安装会提示没有python2。同时使用python --version 会发现commond not found。 所以本…

uniapp小程序单页面改变手机电量,头部通知的颜色效果demo(整理)

onShow(){ // 改变电池的颜色 wx.setNavigationBarColor({ frontColor: ‘#ffffff’, //只支持两种颜色 backgroundColor: ‘#ffffff’, animation: { duration: 1 } }) }

Android 13 - Media框架(9)- NuPlayer::Decoder

这一节我们将了解 NuPlayer::Decoder&#xff0c;学习如何将 MediaCodec wrap 成一个强大的 Decoder。这一节会提前讲到 MediaCodec 相关的内容&#xff0c;如果看不大懂可以先跳过此篇。原先觉得 Decoder 部分简单&#xff0c;越读越发现自己的无知&#xff0c;Android 源码真…

IDEA设置文件编码

IDEA设置文件编码 File->Settings->Editor->File Encodings 均设置为utf-8 新项目 设置 文件编码 点击New Projects Setup 再点击Settings for New Projects File->Settings->Editor->File Encodings 均设置为utf-8

【Linux】文件

Linux 文件 什么叫文件C语言视角下文件的操作文件的打开与关闭文件的写操作文件的读操作 & cat命令模拟实现 文件操作的系统接口open & closewriteread 文件描述符进程与文件的关系重定向问题Linux下一切皆文件的认识文件缓冲区缓冲区的刷新策略 stuout & stderr 什…

windows笔记本远程连接如何打开任务管理器?

参考素材&#xff1a; https://jingyan.baidu.com/article/8275fc86a97f5207a03cf6cd.html https://www.anyviewer.cn/how-to/ctrl-alt-delete-remote-desktop-6540.html 网上查了很多方法&#xff0c;都说ctrlaltend可以解决这个问题。 但是笔记本键盘上没有end键。 继续查了一…