一、一级缓存(本地缓存)
1. 与数据库同一次会话期间,查询到的数据会放在本地缓存中
2. 以后如果需要获取相同的数据,直接从缓存中拿,没必要再去查询数据库
3. sqlSession级别的缓存,一直开启的 —— 实际上就是sqlSession级别的一个Map
一级缓存失效情况(发送多个sql语句):
1. sqlSession不同
2. sqlSession相同,查询条件不同(当前一级缓存中没有这个数据)
3. sqlSession相同,两次查询之间执行了增删改操作(这次增删改操作可能对当前数据有影响)——因为设置了flushCache="true"
4. sqlSession相同,手动清除了一级缓存 openSession.clearCache();
二、二级缓存(全局缓存)
1. 基于NameSpace级别的缓存 —— 一个NameSpace就对应一个二级缓存
2. 工作机制:
* 1)一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中
* 2)如果会话关闭,一级缓存的数据会被保存到二级缓存中;新的会话查询信息,就可以参照二级缓存中的内容
* 3)sqlSession:
* (1) EmployeeMapper ==> Employee
* (2) DepartmentMapper ==> Department
* 不同的NameSpace查出的数据会放在对应的缓存(map)中
使用步骤
* 1. 开启全局二级缓存配置:<setting name="cacheEnabled" value="true"/>
* 2. 在每个Mapper.xml中配置二级缓存
* 3. POJO(每个JavaBean对象)需要实现`Serializable `序列化接口
具体代码:
全局配置文件mybatis-config.xml
<settings>
<setting name="cacheEnabled" value="true"/>
</settings>
cache
标签:二级缓存
eviction
属性:缓存的回收策略
flushInterval
属性:缓存刷新间隔
readOnly
属性:是否只读
true:只读,所有从缓存中获取数据的操作都是只读操作,不会修改数据
为了加快获取速度,直接将数据在缓存中的引用交给用户
—— 不安全,速度快
false:非只读,获取的数据可能被修改,mybatis会利用序列化和反序列化克隆一份新的数据给用户
—— 安全,速度慢
size
属性:缓存中存放多少个元素
type
属性:自定义缓存的全类名 —— 实现Cache接口即可(可以实现第三方缓存整合)
EmployeeMapper.xml
<cache eviction="FIFO" flushInterval="60000" readOnly="false" size="1024"/>
Employee.java
public class Employee implements Serializable {
private static final long serialVersionUID = 1L;
}
注意
在MapperXml配置文件的select
标签不能添加useCache="true" flushCache="true"
设置,否则会刷新缓存
查出的数据会默认先放在一级缓存中,只有关闭会话后,一级缓存中的数据才会转移到二级缓存中
三、和缓存相关的设置/属性:
* 1. <setting name="cacheEnabled" value="true"/>
* 当设置成false时,二级缓存会被关闭,而一级缓存一直可用,不会被关闭
* 2. 每个select标签都有 useCache="true" 设置
* 当设置成false时,二级缓存会被关闭,而一级缓存一直可用,不会被关闭
* 3. 增删改标签: flushCache="true",!!!每次执行完都会清除缓存!!!
* 当设置成true时,一级、二级缓存都会被清除
* 4. 查询标签: flushCache="false"
* 当设置成true时,一级、二级缓存都会被清除
* 5. openSession.clearCache():只是清除当前session的一级缓存
* 6. localCacheScope:本地缓存作用域(一级缓存)
* (1)session:当前会话的所有数据保存在会话缓存中
* (2)statement:可以禁用一级缓存
四、ehcache第三方缓存整合
步骤:
1. 导入第三方缓存包
2. 导入与第三方缓存整合的适配包
3. xxMapper.xml中定义自定义缓存 <cache type="org.mybatis.caches.ehcache.EhcacheCache"/>