1. 简述mybatis缓存机制的实现原理
mybatis缓存分为一级缓存和二级缓存:
一级缓存
- 概念:一级缓存即session缓存,同一个SqlSession查询的内容放在缓存中,供这个SQLSession的其他操作使用,以免产生对数据库的重复操作;mybatis默认一级缓存,但也可以用第三方插件进行配置
- 原理:一个SQLSession对象会使用一个Executor来管理针对当前对话有效的缓存,这个Executor维护了一个Cache对象,该Cache对象用一个Hashmap来保存查询到的数据,map的key为:namespace+statementID、结果范围(offset和limit)、sql语句和查询参数,上述构成一个缓存查询key
- 缓存生命周期:MyBatis在开启一个数据库会话时,会 创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象,Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉。
二级缓存
- 概念:二级缓存即Application缓存,通过配置"cacheEnabled=true"来开启二级缓存
- 原理:MyBatis在为SqlSession对象创建Executor对象时,会对Executor对象加上一个装饰器:CachingExecutor,SqlSession通过这个装饰器对象在Application级别来管理二级缓存,即可以每个mapper都拥有一个独立的cache对象来保存查询数据,也可以多个mapper共享一个cache对象,
- 缓存实现:mybatis自己实现了二级缓存,也可以自己实现cache接口来自定义缓存,也可以使用Memcache第三方缓存库
- 配置缓存:二级缓存配置较麻烦,首先要为每个statement指定是否启用缓存
<select id="selectByMinSalary" resultMap="BaseResultMap" parameterType="java.util.Map" useCache="true">
并且该select语句所在的Mapper,配置了<cache> 或<cached-ref>节点来指定启用哪个Cache对象,并且有效
- 缓存生效顺序:如果同时配置了两级别的缓存,首先查询二级缓存,再查询一级缓存,最后查询数据库
- 缓存实现策略:由于二级缓存是全局的,生命周期跟Application一致,因此合理的缓存保管策略尤为重要,mybatis提供的缓存策略主要包括:LRU(Least Recently Used)、FIFO(First in first out)和Scheduled