2.4Mybatis——缓存机制

server/2024/10/21 7:41:30/

2.4Mybatis——缓存机制

合集总览:Mybatis框架梳理   


讲真,Mybatis缓存这块的记忆已经模糊了。刚好此时写测试用例出现一个BUG,就以这个问题作为切入点来梳理一下。

@Testpublic void test(){Address address = addressMapper.selectById(1L);System.out.println(address);System.out.println("==========================");address = addressMapper.selectById(1L);System.out.println(address);}

问题描述:未进行任何缓存配置,以为会缓存第一次查询结果,但这里控制台打印了两次SQL查。,没有按照预期走Mybatis的缓存策略,分析一下问题出在哪里

缓存配置

官方文档永远是查找问题的首选。关于缓存的介绍,是这样说的:

  1. cacheEnabled 全局性地开启或关闭所有映射器配置文件中已配置的任何缓存,默认为true”

  2. “默认情况下,只启用了本地的会话缓存,它仅仅对一个会话中的数据进行缓存。 要启用全局的二级缓存,只需要在你的 SQL 映射文件中添加一行<cache/>"

总结就是:

  1. 可以全局配置Mybatis的缓存开关,默认开启;
  2. 默认情况下,缓存只在一个sqlSession中生效(即一级缓存开启),同一mapper下(二级缓存)需要手动打开。

所以我们的BUG,如何想让其只执行一次查询,有两种调整方式:

  1. 在AddressMapper.xml中添加<cache/>标签
  2. 两次查询使用同一个sqlSession连接
// 1.使用cache标签
<mapper namespace="org.wyy.mapper.AddressMapper"><cache/><select id="selectById" resultType="org.wyy.dto.Address">select * from address where id = #{id}</select>
</mapper>// 2.使用一个sqlSession连接
@Test
public void test1(){SqlSession sqlSession = sqlSessionFactory.openSession();AddressMapper addressMapper = sqlSession.getMapper(AddressMapper.class);System.out.println(addressMapper.selectById(1L));System.out.println(addressMapper.selectById(1L));sqlSession.close();}

一二级缓存

使用第二种方法正常执行,但使用第一种方法时,提示序列化异常:Cause: java.io.NotSerializableException: org.wyy.dto.Address
这又是什么原因呢?

一级缓存

根据前面几篇文章的描述,我们知道mapper方法的执行分为SQL的映射和SQL的执行,执行阶段又分为执行器构建、statement构建、参数处理、SQL语句执行、结果集处理这几个阶段。
以查询为例,缓存肯定是在执行阶段将查询和结果缓存起来。看一下源码:

@Override
public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {BoundSql boundSql = ms.getBoundSql(parameterObject);// 构建缓存CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql);return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
}public <E> List<E> query(...){// 从本地缓存中获取, localCache是PerpetualCache类型list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;
}public class PerpetualCache implements Cache {private final String id;private Map<Object, Object> cache = new HashMap<>();
}

一级缓存本质是使用Map进行保存,不涉及序列化操作,所以测试中正常执行。

二级缓存

Mybatis默认使用SerializedCache作为二级缓存,由于二级缓存存在持久化和跨会话共享等场景的要求,为了保证缓存对象的完整,被缓存的对象必须实现序列化接口。
除了使用Mybatis默认的二级缓存实现,如果想要自定义二级缓存,只需要实现Mybatis的Cache接口即可,比如通过集成Ehcache或redis来作为Mybatis的缓存实现。

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

所以,Mybatis的缓存设计中,对于仅作用于一次数据库连接内的缓存,使用的是Map集合工具。
对于需要跨线程、跨进程共享的缓存,则是将其存放在JVM实例之外与系统解耦。Mybatis不再关心接入的二级缓存的实现细节,只需实现Cache接口即可。
对于二级缓存而言,官方文档还提供了缓存的容量、驱逐策略等配置属性,感兴趣可以继续探究。


http://www.ppmy.cn/server/128284.html

相关文章

uniapp学习(002 常用的内置组件)

零基础入门uniapp Vue3组合式API版本到咸虾米壁纸项目实战&#xff0c;开发打包微信小程序、抖音小程序、H5、安卓APP客户端等 总时长 23:40:00 共116P 此文章包含第5p-第p10的内容 文章目录 view组件相当于div标签按下松开例子冒泡例子 text组件 相当于span标签scroll-view纵…

利用Spring Boot构建足球青训管理平台

2 相关技术简介 2.1 Java技术 Java是一门伟大的纯面向对象的编程语言和编程语言。同时&#xff0c;它还是Java语言从嵌入式开发到企业级开发的平台。Java凭借其一次编译&#xff0c;任何地方执行的优点&#xff0c;使得盛行的web应用程序有大量的Java编译&#xff0c;很好地支…

信号处理: Block Pending Handler 与 SIGKILL/SIGSTOP 实验

1. 信号处理机制的 “三张表” kill -l &#xff1a;前 31 个信号为系统标准信号。 block pending handler 三张表保存在每个进程的进程控制块 —— pcb 中&#xff0c;它们分别对应了某一信号的阻塞状态、待处理状态以及处理方式。 block &#xff1a;通过 sigset_t 类型实现&…

数据库管理-第245期 主流国产数据库RAC架构概览(20240929)

数据库管理245期 2024-09-29 数据库管理-第245期 主流国产数据库RAC架构概览&#xff08;20240929&#xff09;1 DMDSC2 KingBaseES RAC3 PolarDB4 Cantian5 HaloDB DLB/Data Sharding总结 数据库管理-第245期 主流国产数据库RAC架构概览&#xff08;20240929&#xff09; 作者…

18732 最短路问题

### 思路 1. **建模问题**&#xff1a;将车站和公交线路建模为图&#xff0c;其中车站是节点&#xff0c;公交线路是带权边。 2. **选择算法**&#xff1a;使用Dijkstra算法求解从车站1到车站n的最短路径问题。 3. **初始化**&#xff1a;创建一个优先队列&#xff08;最小堆&…

FinalShell解决Docker日志中文乱码问题

在DockerFile文件末尾添加如下配置即可解决&#xff1a; #解决Docker容器中文显示乱码问题 ENV LANG C.UTF-8 ENV LC_ALL C.UTF-8

php常用的注释符号

如果没有安装vscode和小皮&#xff0c;请点击下方链接安装&#xff1a; Vscode、小皮面板安装-CSDN博客 在学习php过程中&#xff0c;肯定少不了注释&#xff0c;也可以理解为备注的信息&#xff0c;来提醒自己这段代码有什么用&#xff0c;是什么意思等&#xff0c;接下来就介…

平台数据分类与聚类实验报告

参考书籍&#xff1a;《数据流挖掘与在线学习算法》 李志杰 1.6.1 实验目的 本书内容以及课程实验主要涉及Java程序设计语言、数据挖掘工具Weka和数据流机器学习平台MOA&#xff0c;因此&#xff0c;需要安装、配置并熟悉实验环境。Java、Weka和MOA都是开源小软件&#xff0…