MyBatis缓存机制
前言
MyBatis提供了一级缓存和二级缓存的支持。
使用缓存的好处:如果在极短时间内,做相同的查询,那么它们的结果很可能是相同,而访问一次数据库是非常消耗资源的。如果有一层缓存,将极大减少资源的消耗。减少重复的进行数据库的查询
一、一级缓存
一级缓存是SqlSession级别的缓存。默认情况开启一级缓存,基于PerpetualCache的HashMap本地缓存,作用范围为session域内,当session刷新或关闭之后,该session中所有的cache就会被清空。
一级缓存的生命周期
MyBatis在开启一个数据库会话时,会 创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象。Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉。
如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用;
如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用;
SqlSession中执行了任何一个update操作(update()、delete()、insert()) ,都会清空PerpetualCache对象的数据,但是该对象可以继续使用
对于某个查询,根据statementId,params,rowBounds来构建一个key值,根据这个key值去缓存Cache中取出对应的key值存储的缓存结果。如果没有:去数据库中查询数据,得到查询结果;将key和查询到的结果分别作为key,value对存储到Cache中;
mybatis认为,对于两次查询,如果以下条件都完全一样,那么就认为它们是完全相同的两次查询。
传入的statementId
查询时要求的结果集中的结果范围
这次查询所产生的最终要传递给JDBC java.sql.Preparedstatement的Sql语句字符串(boundSql.getSql())
传递给java.sql.Statement要设置的参数值
二、二级缓存
二级缓存就是global caching,它超出了session范围之外,可以被所有SqlSession共享,MyBatis的二级缓存是Application级别的缓存,它可以提高对数据库查询的效率,以提高应用的性能。开启它只需要在MyBatis的核心配置文件(MyBatis-config.xml)settings中设置即可。
二级缓存的使用场景:
二级缓存好处很多,但并不是什么时候都可以使用 。
在以下场景中,推荐使用二级缓存。
以查询为主的应用中,只有尽可能少的增、删、改操作 。
绝大多数以单表操作存在时,由于很少存在互相关联的情况,因此不会出现脏数据 。
可以按业务划分对表进行分组时 , 如关联的表比较少,可以通过参照缓存进行配置。
除了推荐使用的情况,如果脏读对系统没有影响,也可以考虑使用 。 在无法保证数据不出现脏读的情况下 ,不建议使用Mybatis的二级缓存
二级缓存和一级缓存的顺序
你的MyBatis使用了二级缓存,并且你的Mapper和select语句也配置使用了二级缓存,那么在执行select查询的时候,MyBatis会先从二级缓存中取输入,其次才是一级缓存,即MyBatis查询数据的顺序是:
二级缓存 ——> 一级缓存——> 数据库
开启二级缓存
SqlSessionFactory层面上的二级缓存默认是不开启的,二级缓存的开启需要进行配置,实现二级缓存的时候,MyBatis要求返回的POJO必须是可序列化的。 也就是要求实现Serializable接口,配置方法很简单,只需要在映射XML文件(Mapper)配置就可以开启缓存了
映射语句文件中的所有select语句将会被缓存。
映射语句文件中的所欲insert、update和delete语句会刷新缓存。
缓存会使用默认的Least Recently Used(LRU,最近最少使用的)算法来收回。
根据时间表,比如No Flush Interval,(CNFI没有刷新间隔),缓存不会以任何时间顺序来刷新。
缓存会存储列表集合或对象(无论查询方法返回什么)的1024个引用。
缓存会被视为是read/write(可读/可写)的缓存,意味着对象检索不是共享的,而且可以安全的被调用者修改,不干扰其他调用者或线程所做的潜在修改。
mybatis-config.xml配置
<settings>
<setting name="cacheEnabled" value="true"/>
</settings>
mapper.xml配置
<mapper namespace="com.dao.user.UserMapper">
<!--cache配置-->
<!--开启本mapper的namespace下的二级缓存-->
<!--
eviction:代表的是缓存回收策略,目前MyBatis提供以下策略。
(1) LRU,最近最少使用的,一处最长时间不用的对象
(2) FIFO,先进先出,按对象进入缓存的顺序来移除他们
(3) SOFT,软引用,移除基于垃圾回收器状态和软引用规则的对象
(4) WEAK,弱引用,更积极的移除基于垃圾收集器状态和弱引用规则的对象。这里采用的是LRU,
移除最长时间不用的对形象
flushInterval:刷新间隔时间,单位为毫秒,这里配置的是100秒刷新,如果你不配置它,那么当
SQL被执行的时候才会去刷新缓存。
size:引用数目,一个正整数,代表缓存最多可以存储多少个对象,不宜设置过大。设置过大会导致内存溢出。
这里配置的是1024个对象
readOnly:只读,意味着缓存数据只能读取而不能修改,这样设置的好处是我们可以快速读取缓存,缺点是我们没有
办法修改缓存,他的默认值是false,不允许我们修改
-->
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>
……
</mapper>
配置了支持的cache后,如果需要对个别查询进行调整,可以单独设置cache
<select id="getUserList" resultType="User" useCache="true">
</select>