1.概念
将数据库查询结果临时存储在内存中,在后续操作中直接从缓存中获取数据,避免了频繁的数据库查询操作
2.一级缓存
- 一级缓存也叫本地缓存,基于每一个会话sqlsession创建
- 与数据库同一次会话期间查询到的数据会放在本地缓存中,以后如果需要获取相同的数据,可以直接从缓存中取
通过上面的例子可以看到,我们查询了三条数据,但是sql语句只执行了一次,这表明查询出来的数据放在缓存中,下一次查询时直接从缓存中取结果
2.1.缓存失效的情况
- 增删改操作,可能会改变原来的数据,所以必定会刷新缓存
- 查询不同的东西
- 手动清理缓存
结果:
明显看到在清理缓存之后,sql执行了两次
3.二级缓存
3.1.概念
- 二级缓存也叫全局缓存,一级缓存作用域较低,所以有了二级缓存
- 二级缓存是基于namespace的缓存,一个名称空间,对应一个二级缓存
3.2.工作机制
一个会话查询一条数据,这个数据会被放在一级缓存中,如果当这个会话关闭了,这个会话对应的一级缓存就没了,如果开启了二级缓存,一级缓存的数据就会被移到二级缓存中,新的会话查询的数据,就可以从二级缓存中获取
3.3.开启步骤
1.开启全局缓存(全局缓存默认是开启的,但是显式的开启会增加代码的可读性)
<!--显式的开启全局缓存-->
<setting name="cacheEnabled" value="true"/>
2.在Mapper.xml文件中配置
<!--在当前Mapper.xml中使用二级缓存-->
<cache/>
我们还能增加一些参数:
<!--配置二级缓存的参数-->
<cache eviction="FIFO"
flushInterval="60000"
size="512"
readOnly="true"/>
3.测试
结果:
如果没有关闭第一次缓存:
由上面可知,没有关闭第一次会话,二级缓存就不会生效
3.4.注意
- 在<cache/>标签中有一个属性readOnly ,不开启时默认是false,可读写的缓存会(通过序列化)返回缓存对象的拷贝,虽然速度较慢,但是安全
- 因此,如果不开启readOnly,我们必须要序列化实体类,否则会报错,或者我们可以开启readOnly,只读的缓存会给所有调用者返回缓存对象的相同实例,增加了可读性,这样我们不需要序列化实体类