一级缓存
1、什么是一级缓存,为什么使用一级缓存?
每当我们使用MyBatis开启一次和数据库的会话,MyBatis会创建出一个SqlSession对象表示一次数据库会话。
在对数据库的一次会话中,我们有可能会反复地执行完全相同的查询语句,如果不采取一些措施的话,每一次查询
都会查询一次数据库,而我们在极短的时间内做了完全相同的查询,那么它们的结果极有可能完全相同,由于查询一
次数据库的代价很大,这有可能造成很大的资源浪费。
为了解决这一问题,减少资源的浪费,MyBatis会在表示会话的SqlSession对象中建立一个简单的缓存,将每次查 询到的结果结果缓存起来,当下次查询的时候,如果判断先前有个完全一样的查询,会直接从缓存中直接将结果取
出,返回给用户,不需要再进行一次数据库查询了。
缓存示意图:
2、MyBatis如何管理一级缓存
由于MyBatis使用SqlSession对象表示一次数据库的会话,所以,对于会话级别的一级缓存也应该是在SqlSession中控制的。
SqlSession将它的工作交给了Executor执行器这个角色来完成,负责完成对数据库的各种操作。当创建了一个
SqlSession对象时,MyBatis会为这个SqlSession对象创建一个新的Executor执行器,而缓存信息就被维护在这个Executor执行器中,MyBatis将缓存和对缓存相关的操作封装成了Cache接口中。
SqlSession、Executor、Cache之间的关系如下列类图所示(一级缓存PerpetualCache类型的永久缓存,保存在执行器中(BaseExecutor),而执行器又在SqlSession中,所以一级缓存的生命周期与SqlSession是相同的。
):
PerpetualCache实现原理:内部维护一个HashMap,所有的缓存操作,其实都是对这个HashMap的操作。
3、一级缓存的生命周期
- MyBatis在开启一个数据库会话时,会创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象,Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉;
- 如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用;
- 如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用;
- SqlSession中执行了update()、delete()、insert()操作,都会清空PerpetualCache对象的数据,但是该对象可以继续使用;
4、一级缓存的工作流程
- 对于某个查询,根据statementId,params,rowBounds来构建一个key值,根据这个key值去缓存Cache中取出对应的key值存储的缓存结果;
- 判断从Cache中根据特定的key值取的数据数据是否为空,即是否命中;如果命中,则直接将缓存结果返回;
- 如果没命中:
- 去数据库中查询数据,得到查询结果;
- 将key和查询到的结果分别作为key,value对存储到Cache中;
- 将查询结果返回;
二级缓存
1、二级缓存机制
- 当开一个会话时,一个SqlSession对象会使用一个Executor对象来完成会话操作,MyBatis的二级缓存机制的关键就是对这个Executor对象做文章。
- 如果用户配置了"cacheEnabled=true",会创建CachingExecutor对象,这时SqlSession使用CachingExecutor对象来完成操作请求。
- CachingExecutor对于查询请求,会先判断该查询请求在二级缓存中是否有缓存结果,如果有查询结果,则直接返回缓存结果;如果缓存中没有,再交给真正的Executor对象来完成查询操作,之后CachingExecutor会将真正Executor返回的查询结果放置到缓存中,然后在返回给用户。
2、二级缓存的创建
为每一个Mapper分配一个Cache缓存对象(使用节点配置):
这个配置创建了一个 FIFO 缓存,并每隔 60 秒刷新,存数结果对象或列表的 512 个引用,而且返回的对象被认为是只
读的,因此在不同线程中的调用者之间修改它们会导致冲突。
多个Mapper共用一个Cache缓存对象(使用节点配置):
使用前提:
- MyBatis支持二级缓存的总开关:全局配置变量参数 cacheEnabled=true;
- 该select语句所在的Mapper,配置了
或节点,并且有效; - 该select语句的参数 useCache=true;
cache属性:
- eviction:
LRU – 最近最少使用的:移除最长时间不被使用的对象。
FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
SOFT – 软引用:移除基于垃圾回收器状态和软引用规则的对象。
WEAK – 弱引用:更积极地移除基于垃圾收集器状态和弱引用规则的对象。 - flushInterval:刷新间隔
- size:要缓存的对象的数目
- readOnly:设置对象的只读性