Mybatis进阶使用-一级缓存与二级缓存
简介
缓存是一般的ORM 框架都会提供的功能,目的就是提升查询的效率和减少数据库的压力。跟Hibernate 一样,MyBatis 也有一级缓存和二级缓存,并且预留了集成第三方缓存的接口。
一级缓存
- 什么是一级缓存?
Mybatis一级缓存实际上就是一个依赖于SqlSession的缓存对象,PerpetualCache里面的结构很简单,通过一个k-v结构的cache维护缓存数据。一级缓存默认开启
public class PerpetualCache implements Cache {
private final Strng id;
private Map<Object, Object> cache = new HashMap<Object, Object>();
一级缓存的生命周期?
PerpetualCache的生命周期是和SqlSession相关的,即只有在同一个SqlSession中,一级缓存才会用到。- 如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用;
- 如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用;
- SqlSession中执行了任何一个update操作(update()、delete()、insert()) ,都会清空PerpetualCache对象的数据,但是该对象可以继续使用;
开发时,如何才能真正用到一级缓存?
我们经常在某个方法中进行多次数据库查询,在实际场景中,每次的数据库查询都会开启一个新的会话(SqlSession)。这种情况下我们是没有用到一级缓存的,因为根本就没有复用到SqlSession。
那么我们怎样控制程序复用SqlSession,使get()能用到一级缓存呢?
其中一种办法就是开启一个事务。
二级缓存
- 什么是二级缓存?
二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。二级缓存 底层还是 HashMap 架构。 - 二级缓存的生命周期?
- 映射语句中的所有select语句将会被缓存
- 映射语句中的所有insert,update和delete语句会刷新缓存。
- 缓存会使用默认的Latest Recently Used(LRU,最近最少使用的)算法来回收
- 根据时间表,比如No Flush Interval,(CNFI,没有刷新间隔),缓存不会以任何时间顺序来刷新。
- 缓存会存储列表集合或对象(无论查询方法返回什么)的1024个引用
- 缓存会被视为是read/write(可读/可写)的缓存,意味着对象检索不是共享的,而且可以安全的被调用者修改,不干扰其他调用者或者线程所做的潜在修改。
- 如何才能用到二级缓存?
二级缓存默认是不开启的,需要手动开启二级缓存,实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的,否则会抛出异常。开启二级缓存的条件也是比较简单,通过直接在 MyBatis 配置文件中通过
<setting name="cacheEnabled" value="true"/> <!-- 二级缓存开启 -->
或者在springboot项目配置文件中增加如下配置项:
mybatis.configuration.cache-enabled=true
来开启二级缓存,还需要在 Mapper 的xml 配置文件中加入 标签
设置 cache 标签的属性
cache 标签有多个属性,一起来看一些这些属性分别代表什么意义eviction: 缓存回收策略,有这几种回收策略
- LRU - 最近最少回收,移除最长时间不被使用的对象
- FIFO - 先进先出,按照缓存进入的顺序来移除它们
- SOFT - 软引用,移除基于垃圾回收器状态和软引用规则的对象
- WEAK - 弱引用,更积极的移除基于垃圾收集器和弱引用规则的对象
- 默认是 LRU 最近最少回收策略
flushinterval 缓存刷新间隔,缓存多长时间刷新一次,默认不清空,设置一个毫秒值
readOnly: 是否只读;true 只读,MyBatis 认为所有从缓存中获取数据的操作都是只读操作,不会修改数据。MyBatis 为了加快获取数据,直接就会将数据在缓存中的引用交给用户。不安全,速度快。读写(默认):MyBatis 觉得数据可能会被修改
size : 缓存存放多少个元素
type: 指定自定义缓存的全类名(实现Cache 接口即可)
blocking: 若缓存中找不到对应的key,是否会一直blocking,直到有对应的数据进入缓存。
参考
结语
欢迎关注微信公众号『码仔zonE』,专注于分享Java、云计算相关内容,包括SpringBoot、SpringCloud、微服务、Docker、Kubernetes、Python等领域相关技术干货,期待与您相遇!
最新文章
- linux 正则表达式使用
- Bash之PS1配置
- EF架构~关系表插入应该写在事务里,但不应该是分布式事务
- 关于Android Force Close 出现的原因 以及解决方法
- DP~数塔(hrbustoj1004)
- System.Net.Sockets.Socket SendAsync System.ObjectDisposedException: Cannot access a disposed object.
- 161101、在Java中如何高效判断数组中是否包含某个元素
- nvl
- 利用WHID为隔离主机建立隐秘通道
- 高级开发层面,针对Hibernate方面面试题的总结(对其它ORM也适用)
- 『2019/3/8 USACO测试 反思与总结』
- nginx1.14.0版本location路径配置四种方式
- UVA 12293 - Box Game(博弈)
- Python+OpenCV图像处理(三)—— Numpy数组操作图片
- find命令归总
- git版本管理工具-git的概述
- dubbo接口访问控制
- jaxb教程(忘记了过来看看)
- 嵌入式Linux的web视频服务器的构建
- sql server中带有output的DML
热门文章
- three.js 欧拉角和四元数
- PHP入门之流程控制
- List接口(动态数组)
- python map() filter() reduce()函数的用法以及实例
- 用大白话讲Java动态代理的原理
- 字节数组X中存放着 0~F共16个十六进制数,请将这些数以十六进制形式显示在屏幕上。
- PHP sizeof() 函数
- UOJ 422 [集训队作业2018] 小Z的礼物 min-max容斥 期望 轮廓线dp
- 海华大赛第一名团队聊比赛经验和心得:AI在垃圾分类中的应用
- 小程序中 radio 的一个坑,到底 checked 该赋什么值?