简介
缓存是一般的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,直到有对应的数据进入缓存。
-
参考
- https://blog.csdn.net/n950814abc/article/details/97611712
结语
欢迎关注微信公众号『码仔zonE』,专注于分享Java、云计算相关内容,包括SpringBoot、SpringCloud、微服务、Docker、Kubernetes、Python等领域相关技术干货,期待与您相遇!