MyBatis缓存机制
1、缓存机制简介
用户发送请求,dispatchServlet会根据请求调用对应的Controller进行处理,然后经过Service和Mapper,当Mapper去数据库查询数据之前,会先去缓存里去找,如果在缓存里没有找到,再去数据库里查询,这样可以避免和数据库的多次交互,提高效率。
-
MyBatis 包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制,缓存可以极大的提升查询效率。
-
MyBatis系统中默认定义了两级缓存
- 一级缓存
- 二级缓存
-
默认情况下,只有一级缓存(SqlSession级别的缓存,也称为本地缓存)开启。
-
二级缓存需要手动开启和配置,它是基于namespace级别的缓存。
-
为了提高扩展性。MyBatis定义了缓存接口Cache。我们可以通过实现Cache接口来自定义二级缓存。
-
缓存查找顺序:二级缓存 -> 一级缓存 -> 数据库
2、一级缓存的使用
-
一级缓存(local cache),即本地缓存,作用域默认为sqlSession。当 Session flush 或 close 后,该 Session 中的所有 Cache 将被清空。
-
本地缓存不能被关闭,但可以调用 clearCache() 来清空本地缓存,或者改变缓存的作用域。
-
在mybatis3.1之后,可以配置本地缓存的作用域。在 mybatis.xml 中配置:
设置 | 说明 | 可用值 | 默认值 |
---|---|---|---|
localCacheScope | MyBatis利用本地缓存机制(Local Cache)防止循环引用和加速重复嵌套查询。默认值为SESSION,这种情况下会缓存一个会话中执行的所有查询。若设置为STATEMENT,本地会话仅用在语句执行上,对相同SqlSession的不同调用将不会共享数据。 | SESSION|STATEMENT | SESSION |
- 一级缓存的工作机制:同一次会话期间只要查询过的数据都会保存在当前SqlSession的一个Map中。
5)原理:当使用SqlSession进行查询操作的时候,首先去缓存中中查询,如果缓存中没有,那么去数据库中查找数据。数据库查到数据之后,返回数据,并且向一级缓存中保存一份数据,下次再查询的时候直接从缓存中获取。
key: hashCode+查询的SqlId+编写的sql查询语句+参数
3、一级缓存失效的几种情况
-
不同的SqlSession对应不同的一级缓存
-
同一个SqlSession但是查询条件不同
-
同一个SqlSession两次查询期间执行了任何一次增删改操作
-
同一个SqlSession两次查询期间手动清空了缓存
4、二级缓存的使用
二级缓存,全局范围缓存,基于namespace级别的缓存,基于同一个namespace的多个SqlSession对象共享同一个二级缓存。
1)二级缓存使用的步骤:
① 全局配置文件中开启二级缓存
<!--二级缓存默认是关闭的,需要手动开启-->
<setting name="cacheEnabled" value="true"/>
② 需要使用二级缓存的映射文件处使用cache配置缓存
<!--
eviction=“FIFO”:缓存回收策略
LRU – 最近最少使用的:移除最长时间不被使用的对象。
FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。
默认的是 LRU。
-->
<cache eviction="LRU" />
③ 注意:POJO需要实现Serializable接口
2)二级缓存工作原理:基于同一个namespace的多个SqlSession对象,第一次查询,先从 缓存中找,如果找不到,则从数据库中查找,将从数据库中查找到的数据存储到当前SqlSession对象的一级缓存中,只有当SqlSession对象进行关闭或者提交操作的时候,才会把一级缓存中的数据存放到二级缓存中。