第一步:在mybatis-config.xml 中配置了(可以不配置,默认是true):
<setting name="cacheEnabled" value="true"/>
只要没有显式地设置cacheEnabled=false,都会用CachingExecutor 装饰基本的执行器。
第二步:在Mapper.xml 中配置<cache/>标签:
<!-- 声明这个namespace 使用二级缓存-->
<cache type="org.apache.ibatis.cache.impl.PerpetualCache"
size="1024" <!—最多缓存对象个数,默认1024-->
eviction="LRU" <!—回收策略-->
flushInterval="120000" <!—自动刷新时间ms,未配置时只有调用时刷新-->
readOnly="false"/> <!—默认是false(安全),改为true 可读写时,对象必须支持序列
化-->
cache 属性详解:
属性 | 含义 | 取值 |
---|---|---|
type | 缓存实现类 | 需要实现Cache 接口,默认是PerpetualCache |
size | 最多缓存对象个数 | 默认1024 |
eviction | 回收策略(缓存淘汰算法) | LRU – 最近最少使用的:移除最长时间不被使用的对象(默认)。 FIFO – 先进先出:按对象进入缓存的顺序来移除它们。 |
flushInterval | 定时自动清空缓存间隔 | 自动刷新时间,单位ms,未配置时只有调用时刷新 |
readOnly | 是否只读 | true:只读缓存;会给所有调用者返回缓存对象的相同实例。因此这些对象 不能被修改。这提供了很重要的性能优势。 false:读写缓存;会返回缓存对象的拷贝(通过序列化),不会共享。这 会慢一些,但是安全,因此默认是false。 改为false 可读写时,对象必须支持序列化。 |
blocking | 是否使用可重入锁实现 缓存的并发控制 | true,会使用BlockingCache 对Cache 进行装饰 默认false |
Mapper.xml 配置了<cache>之后,select()会被缓存。update()、delete()、insert()会刷新缓存。
思考:如果cacheEnabled=true,Mapper.xml 没有配置标签,还有二级缓存吗?还会出现CachingExecutor 包装对象吗?
只要cacheEnabled=true 基本执行器就会被装饰。有没有配置<cache>,决定了在启动的时候会不会创建这个mapper 的Cache 对象,最终会影响到CachingExecutorquery 方法里面的判断:
if (cache != null) {
如果某些查询方法对数据的实时性要求很高,不需要二级缓存,怎么办?
我们可以在单个Statement ID 上显式关闭二级缓存(默认是true):
<select id="selectBlog" resultMap="BaseResultMap" useCache="false">
了解了二级缓存的工作位置和开启关闭的方法之后,我们也来验证一下二级缓存。