一、一级缓存
MyBatis 的一级缓存存在于 SqlSession 的生命周期中,在同一个 SqlSession 中查询时,MyBatis 会把执行的方法和参数通过算法生成缓存的键值,将键值和查询结果存入一个 Map对象中。如果同一个 SqlSession 中执行的方法和参数完全一致,那么通过算法会生成相同的键值,当 Map 缓存对象中己经存在该键值时,则会返回缓存中的对象。
SysUser user1 = userMapper.selectByid(lL);
user1.setUserName("New Name");
SysUser user2 = userMapper.selectByid(lL);
当前只执行一次数据库查询,如果对user1重新赋值,对应user2查询的对象值也修改,当前查询出user2的username为“New Name”。
如果想要去掉一级缓存(不建议),可以设置属性flushCache="true",如下
【说明】任何的 INSERT 、UPDATE 、 DELETE 操作都会清空一级缓存,在执行过INSERT、UPDATE、DELETE方法之后查询都会重新查询数据库。
二、二级缓存
MyBatis的二级缓存非常强大,它不同于一级缓存只存在于 SqlSession 的生命周期中,而是可以理解为存在于 SqlSessionFactory 的生命周期中。
配置二级缓存
在 MyBatis 的全局配置 settings 中有一个参数 cacheEnabled,这个参数是二级缓存的全局开关,默认值是 true,初始状态为启用状态。如果把这个参数设置为false,即使有后面的二级缓存配置,也不会生效 。
【说明】默认状态下二级缓存即为启用状态,可不配置。
MyBatis的二级缓存是和命名空间绑定的,即二级缓存需要配置在Mapper. xml 映射文件中,或者配置在 Mapper.java 接口中。在映射文件中,命名空间就是 XML 根节点mapper的namespace属性。在 Mapper 接口中,命名空间就是接口的全限定名称。
2.1 Mapper.xml配置
配置<cache />即可,如下:
默认的二级缓存会有如下效果:
- 映射语句文件中的所有 SELECT 语句将会被缓存。
- 映射语句文件中的所有INSERT、UPDATE、DELETE 语句会刷新缓存。
- 缓存会使用 Least Recently Used ( LRU,最近最少使用的)算法来收回。
- 根据时间表(如 no Flush Interval ,没有刷新间隔),缓存不会以任何时间顺序来刷新。
- 缓存会存储集合或对象(无论查询方法返回什么类型的值)的 1024 个引用。
- 缓存会被视为 read/write (可读/可写)的,意味着对象检索不是共享的,而且可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。
所有的这些属性都可以通过缓存元素的属性来修改,示例如下 。
<cache
eviction="FIFO"
flushlnterval="60000"
size="512"
readOnly="true"/>
属性介绍:
eviction(收回策略)
LRU(最近最少使用的):移除最长时间不被使用的对象,这是默认值。
FIFO(先进先出):按对象进入缓存的顺序来移除它们。
SOFT(软引用):移除基于垃圾回收器状态和软引用规则的对象。
WEAK(弱引用):更积极地移除基于垃圾收集器状态和弱引用规则的对象。
flushinterval(刷新间隔)
可以被设置为任意的正整数,而且它们代表一个合理的毫秒形式的时间段。默认情况不设置,即没有刷新间隔,缓存仅仅在调用语句时刷新。
size(引用数目)
可以被设置为任意正整数,要记住缓存的对象数目和运行环境的可用内存资源数目。默认值是1024 。
readOnly(只读)
属性可以被设置为 true 或 false。只读的缓存会给所有调用者返回缓存对象的相同实例,因此这些对象不能被修改,这提供了很重要的性能优势。
可读写的缓存会通过序列化返回缓存对象的拷贝,这种方式会慢一些,但是安全,因此默认是 false。
2.2 Mapper接口配置
在使用注解方式时,如果想对注解方法启用二级缓存,还需要在Mapper接口中进行配置,如果Mapper接口也存在对应的XML映射文件,两者同时开启缓存时,还需要特殊配置。
当只使用注解方式配置二级缓存时,在接口中增加如下配置:
@CacheNamespace(
eviction = FifoCache.class,
flushinterval = 60000,
size = 512,
readWrite = true
)
当同时使用注解方式和 XML 映射文件时,Mapper 接口和对应的 XML 文件是相同的命名空间,想使用二级缓存,两者必须同时配置,这个时候应该使用参照缓存,如下
@CacheNamespaceRef(RoleMapper.class)
public interface RoleMapper{
}
因为想让 RoleMapper 接口中的注解方法和 XML 中的方法使用相同的缓存,因此使用参照缓存配置RoleMapper.class,这样就会使用命名空间为 com.chl.mapper.RoleMapper的缓存配置,即 RoleMapper.xml 中配置的缓存。
Mapper 接口可以通过注解引用 XML 映射文件或者其他接口的缓存,在 XML 中也可以配置参照缓存,如可以在 RoleMapper.xml 中进行如下修改
<cache-ref narnespace=" com.chl.mapper.RoleMapper"/>
这样配置后,XML 就会引用 Mapper 接口中配置的二级缓存,同样可以避免同时配置二级缓存导致的冲突 。
配置如下:
2.3 缓存使用实例
因为使用可读写缓存,可以使用 SerializedCache 序列化缓存。这个缓存类要求所有被序列化的对象必须实现 Serializable (java.io.Serializable)接口。
修改SysRole.java
修改RoleMapper.java
修改RoleMapper.xml
通过JUnit测试
日志输出
日志中存在好几条以 Cache Hit Ratio 开头的语句 ,这行日志后面输出的值为当前执行方法的缓存命中率。