mybatis 一级缓存是啥?5 个面试题回答你
一级缓存
主要内容:
data:image/s3,"s3://crabby-images/b74ae/b74ae86f4517894b72139030dfad8526095d0a7e" alt=""
一级缓存也叫本地缓存(LocalCache),Mybatis的一级缓存是会话级别(SqlSession)层面进行缓存的。Mybatis的一级缓存是默认开启的。我们开发项目中不需要做任何配置,但是如果想关闭一级缓存,可以使用localCacheScopde=statement来关闭。
如何关闭一级缓存呢?
在BaseExecutor的中,请看下面代码:
data:image/s3,"s3://crabby-images/90b8e/90b8e079b4166cc11fe0f0f49db495fc74f0c085" alt=""
为什么说是SqlSession层面缓存?
就是一级缓存的生命周期和一个SqlSession对象的生命周期一样。
下面这段中,就会使用到一级缓存。
`SqlSession sqlSession1 = sqlSessionFactory.openSession();
User user1 = sqlSession1.selectOne("com.tian.mybatis.mapper.UserMapper.selectUserById", 1);
User user2 = sqlSession1.selectOne("com.tian.mybatis.mapper.UserMapper.selectUserById", 1);
`
结果输出:
data:image/s3,"s3://crabby-images/74301/743016e8c0da356e148a9fc89dd3f9a62d91cc83" alt=""
用两张图来总结:
第一次:查数据库,放入到缓存中。
data:image/s3,"s3://crabby-images/9037b/9037b221e56ebef2722942defe890333b5735737" alt=""
第二次:直接从缓存中获取。
data:image/s3,"s3://crabby-images/309bd/309bd7a7605b0a68f17ace42dc98cebe295c7029" alt=""
下面这段代码中就使用不到缓存
`SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
sqlSession = sqlSessionFactory.openSession();
sqlSession1 = sqlSessionFactory.openSession();
UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class);
System.out.println("第一次查询");
System.out.println(userMapper.selectById(1));
User user = new User();
user.setUserName("tian111");
user.setId(1);
userMapper1.updateAuthorIfNecessary(user);
System.out.println("第二次查询");
System.out.println(userMapper.selectById(1));
`
输出结果:
data:image/s3,"s3://crabby-images/2d23a/2d23a6ed0b92e060e0eab206b615866ee7719d17" alt=""
用三张图来总结:
第一次查询:sqlSession1查询数据库,放入到缓存中。
data:image/s3,"s3://crabby-images/e0aef/e0aef90f85a0caafc07001f4f957df8980fa9984" alt=""
更新:sqlSession2进行更新,注意这里写入的是sqlSession自己的本地缓存。
data:image/s3,"s3://crabby-images/009b0/009b0f462ae80ae469794905b04e299807f77aa5" alt=""
第二次查询:sqlSession1第二次查询。
data:image/s3,"s3://crabby-images/9a992/9a99288f498a856c96a94cd58cc603e19ba2656e" alt=""
记住是一级缓存只能是同一个SqlSession对象就行了。
一级缓存维护在哪里的?
既然一级缓存的生命周期和SqlSession一致,那么我们可以猜想,这个缓存是不是就维护在SqlSession中呢?
SqlSession的默认实现类DefaultSqlSession,在DefaultSqlSession中只有两个属性可能存放缓存:
`private final Configuration configuration;
private final Executor executor;
`
configuration是全局的,肯定不会放缓存。
那就只能把希望寄托于Executor了。由于Executor是个接口,我们可以看看他的实现类:
data:image/s3,"s3://crabby-images/04766/04766b643743103be750a1ba2978e6b72d992fb8" alt=""
另外这里有个BaseExecutor。有各类也得瞄瞄。一看居然有东西。
`public abstract class BaseExecutor implements Executor {
private static final Log log = LogFactory.getLog(BaseExecutor.class);
protected Transaction transaction;
protected Executor wrapper;
protected ConcurrentLinkedQueue<DeferredLoad> deferredLoads;
//熟悉的家伙,基本缓存
protected PerpetualCache localCache;
protected PerpetualCache localOutputParameterCache;
protected Configuration configuration;
protected int queryStack;
private boolean closed;
protected BaseExecutor(Configuration configuration, Transaction transaction) {
this.transaction = transaction;
this.deferredLoads = new ConcurrentLinkedQueue<>();
this.localCache = new PerpetualCache("LocalCache");
this.localOutputParameterCache = new PerpetualCache("LocalOutputParameterCache");
this.closed = false;
this.configuration = configuration;
this.wrapper = this;
}
}
`
再看看BaseExecutor类图:
data:image/s3,"s3://crabby-images/2c59c/2c59c4ece682d69a391aa5220ab949349c666ccf" alt=""
所以这就证明了,这个缓存是维护在SqlSession里。
一级缓存什么时候被清空?
在执行update、insert、delete、flushCache="true"、commit、rollback、LocalCacheScope.STATEMENT等情况下,一级缓存就都会被清空。
`@Override
public void clearLocalCache() {
if (!closed) {
localCache.clear();
localOutputParameterCache.clear();
}
}
`
update时,一级缓存会被清空。delete和insert都是调用这个update。可以从SqlSession的insert、update、delete方法跟踪。
data:image/s3,"s3://crabby-images/ff336/ff336f8d1704a302d5b14db2ece572a5be22a23f" alt=""
LocalCacheScope.STATEMENT时,一级缓存会被清空。在BaseExecutor里的query方法中:
data:image/s3,"s3://crabby-images/05757/05757f08e0374ee9189df6fea8473d784319dbf3" alt=""
事务提交回滚时,一级缓存会被清空。
data:image/s3,"s3://crabby-images/e0eed/e0eed9f47aedd91ec896c5cdb712c76cd32b8f24" alt=""
flushCache="true"时,一级缓存会被清空。
data:image/s3,"s3://crabby-images/6eaa2/6eaa23b248574ce5b5bdddc8d9461451004659f1" alt=""
一级缓存key是什么?
data:image/s3,"s3://crabby-images/cdafa/cdafadf2f1c564269ff6167eea3d6b58cba370e3" alt=""
下面就是一级缓存key的创建过程
`@Override
public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {
if (closed) {
throw new ExecutorException("Executor was closed.");
}
CacheKey cacheKey = new CacheKey();
cacheKey.update(ms.getId());
cacheKey.update(rowBounds.getOffset());
cacheKey.update(rowBounds.getLimit());
cacheKey.update(boundSql.getSql());
List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();
// mimic DefaultParameterHandler logic
for (ParameterMapping parameterMapping : parameterMappings) {
if (parameterMapping.getMode() != ParameterMode.OUT) {
Object value;
String propertyName = parameterMapping.getProperty();
if (boundSql.hasAdditionalParameter(propertyName)) {
value = boundSql.getAdditionalParameter(propertyName);
} else if (parameterObject == null) {
value = null;
} else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
value = parameterObject;
} else {
MetaObject metaObject = configuration.newMetaObject(parameterObject);
value = metaObject.getValue(propertyName);
}
cacheKey.update(value);
}
}
if (configuration.getEnvironment() != null) {
// issue #176
cacheKey.update(configuration.getEnvironment().getId());
}
return cacheKey;
}
`
id:com.tian.mybatis.mapper.UserMapper.selectById
key的生成策略:id + offset + limit + sql + param value + environment id,这些值都相同,生成的key就相同。
示例:
data:image/s3,"s3://crabby-images/f7a92/f7a9264f03e951c62acff365b6a694c0f9f8b6c2" alt=""
一级缓存总结
一级缓存的生命周期和SqlSession对象的生命周期一致。所以缓存维护在SqlSession中的属性executor里。
一级缓存默认开启。可以通过修改配置项把一级缓存关掉。
清空一级缓存的方式有:
update、insert、delete
flushCache="true"
commit、rollback
LocalCacheScope.STATEMENT
评论