Mybatis一级缓存原理
来源:互联网 发布:带着淘宝去异界txt 编辑:程序博客网 时间:2024/05/17 23:55
一级缓存概念
当我们使用Mybatis进行数据库的操作时候,会创建一个SqlSession来进行一次数据库的会话,会话结束则关闭SqlSession对象。那么一个SqlSession的生命周期即对应于Mybatis的一次会话。在Mybatis的一次会话中,我们很有可能多次查询完全相同的sql语句,如果不采取措施的话,每一次查询都查询一次数据库。而一次会话时间一般都是极短的,相同Sql的查询结果极有可能完全相同。由于查询数据库代价是比较大的,这会导致系统的资源浪费。
为了解决这个问题,Mybatis对每一次会话都添加了缓存操作。这个缓存的作用域为一次会话中。缓存随着会话(SqlSession)的创建而产生,随着会话结束而释放。对一次会话的查询操作,总是先查看缓存中是否存在查询结果,如果存在则直接取缓存中的结果,不存在则查询数据库。这样的话,一次会话中的完全相同的查询则只会查询一次,节省了系统资源。
一级缓存的实现
我们知道,对SqlSession的操作mybatis内部都是通过Executor来执行的。Executor的生命周期和SqlSession是一致的。Mybatis在Executor中创建了本地缓存(一级缓存)。如下图:
下面我们对照着Mybatis的源码看下具体的实现,先看一级缓存对象的创建。我们知道所有的Mybatis提供的三个Executor实现类都继承了BaseExecutor。在Executor创建(SimpleExecutor)时候会调用父类的初始化方法。先看BaseExecurot的构造方法。
protected BaseExecutor(Configuration configuration, Transaction transaction) { this.configuration = configuration; this.transaction = transaction; this.deferredLoads = new ConcurrentLinkedQueue<DeferredLoad>(); this.closed = false; this.wrapperExecutor = this; //mybatis一级缓存,在创建SqlSession->Executor时候动态创建,随着sqlSession销毁而销毁 this.localCache = new PerpetualCache("LocalCache"); this.localOutputParameterCache = new PerpetualCache("LocalOutputParameterCache");}
我们可以看到一级缓存的实现很简单,不能像二级缓存那样设置淘汰规则过期时间等等,采用PerpetualCache作为实现类,底层使用HashMap存储(源码略)。
缓存只对我们的查询有效,对数据库写和更新删除是无效的,我们继续看下Executor中是怎么使用缓存的。具体为Executor接口的query方法实现.
//SqlSession.selectList会调用此方法(一级缓存操作,总是先查询一级缓存,缓存中不存在再查询数据库)public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException { ErrorContext.instance().resource(ms.getResource()).activity("executing a query").object(ms.getId()); if (closed) {//如果已经关闭,报错 throw new ExecutorException("Executor was closed."); } //先清一级缓存,再查询,但仅仅查询堆栈为0才清,为了处理递归调用 if (queryStack == 0 && ms.isFlushCacheRequired()) { clearLocalCache(); } List<E> list; try { //加一,这样递归调用到上面的时候就不会再清局部缓存了 queryStack++; list = resultHandler == null ? (List<E>) localCache.getObject(key) : null; if (list != null) { //如果查到localCache缓存,处理localOutputParameterCache handleLocallyCachedOutputParameters(ms, key, parameter, boundSql); } else { //从数据库查 list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql); } } finally { queryStack--; //清空堆栈 } if (queryStack == 0) { //延迟加载队列中所有元素 for (DeferredLoad deferredLoad : deferredLoads) { deferredLoad.load(); } deferredLoads.clear(); //清空延迟加载队列 if (configuration.getLocalCacheScope() == LocalCacheScope.STATEMENT) { clearLocalCache(); } } return list;}
通过源码可以看到,Executor在执行数据库查询的时候总是先查看缓存中是否存在,若不存在则查询数据库。
一级缓存生命周期
- MyBatis在开启一个会话时,会创建一个新的SqlSession对象,SqlSession对象中会有一个新的Executor对象,Executor对象中持有一个新的PerpetualCache对象;当会话结束时,SqlSession对象及其内部的Executor对象还有PerpetualCache对象也一并释放掉。
- 如果SqlSession调用了close()方法,会释放掉一级缓存PerpetualCache对象,一级缓存将不可用;
- 如果SqlSession调用了clearCache(),会清空PerpetualCache对象中的数据,但是该对象仍可使用;
- SqlSession中执行了任何一个update操作(update()、delete()、insert()),都会清空PerpetualCache对象的数据,但是该对象可以继续使用;
一级缓存注意事项
- MyBatis对会话(Session)级别的一级缓存设计的比较简单,就简单地使用了HashMap来维护,并没有对HashMap的容量和大小进行限制。
- 一般而言SqlSession的生存时间很短。一般情况下使用一个SqlSession对象执行的操作不会太多,执行完就会消亡;
- 对于某一个SqlSession对象而言,只要执行update操作(update、insert、delete),都会将这个SqlSession对象中对应的一级缓存清空掉,所以一般情况下不会出现缓存过大,影响JVM内存空间的问题;
- 可以手动地释放掉SqlSession对象中的缓存。
- 一级缓存是一个粗粒度的缓存,没有更新缓存和缓存过期的概念
- 对于数据变化频率很大,并且需要高时效准确性的数据要求,我们使用SqlSession查询的时候,要控制好SqlSession的生存时间,SqlSession的生存时间越长,它其中缓存的数据有可能就越旧,从而造成和真实数据库的误差;同时对于这种情况,用户也可以手动地适时清空SqlSession中的缓存;
- 对于只执行、并且频繁执行大范围的select操作的SqlSession对象,SqlSession对象的生存时间不应过长。
如何禁用一级缓存
我们知道,mybatis的一级缓存是内部实现的一个特性,用户不能配置,默认情况下框架自动支持缓存。那万一业务场景下需要禁用一级缓存怎么操作呢?我们可以使用Mybatis的插件开发来做。
@Intercepts({@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})// 禁用Mybatis一级缓存拦截器public class CloseLocalCacheInterceptor implements Interceptor { public Object intercept(Invocation invocation) throws Throwable { if (invocation.getTarget() instanceof Executor) { Executor executor = (Executor) invocation.getTarget(); executor.clearLocalCache(); } return invocation.proceed(); } public Object plugin(Object target) { return Plugin.wrap(target, this); } public void setProperties(Properties properties) { }}
- Mybatis一级缓存原理
- Mybatis一级缓存原理
- 【MyBatis框架】查询缓存-一级缓存原理
- 【MyBatis框架】查询缓存-一级缓存原理
- mybatis的一级缓存实现原理
- mybatis原理-mybatis的一级缓存详解和注意事项
- MyBatis一级缓存
- Mybatis一级缓存
- Mybatis一级缓存
- MyBatis 一级缓存实现
- mybatis的一级缓存
- MyBatis-一级缓存
- mybatis一级缓存
- mybatis中一级缓存
- mybatis一级缓存
- mybatis的一级缓存
- MyBatis的一级缓存
- MyBatis一级缓存,二级缓存
- 读/写视频文件
- Android动画
- 476. Number Complement
- 剑指offer——数组中出现次数超过一半的数字
- 归并排序
- Mybatis一级缓存原理
- C编程(五)流程控制之循环while,do while
- 【题解】 NOIP2012pj 寻宝 洛谷P1076
- DWA算法
- adb操作命令详解及大全
- 0707学习总结(类,namespace,函数重载,带默认值得函数)
- multiset
- Java大数据学习路线图
- 最优装载问题