likes
comments
collection
share

水煮MyBatis(十二)- 一级缓存

作者站长头像
站长
· 阅读数 58

前言

缓存是Mybatis里比较有意思的一个特性,一定程度上可以提高查询效率,降低数据库I/O压力。应对的场景是这样:在短时间内,频繁的反复执行相同的查询语句,如果任由其调用数据库,会对系统性能造成负面影响,所以缓存机制就出现了。分了两个层级,会话级和命名空间级别,这一章先介绍会话级缓存。

缓存类图

水煮MyBatis(十二)- 一级缓存

从上面这个图中,能看到PerpetualCache类是被独立列出来的,有三个原因:

  • PerpetualCache是会话级【一级】缓存的默认实现,注意:一级缓存只会用到PerpetualCache
  • Cache的其他实现,比如soft、weak等,都是PerpetualCache的一个封装;
  • Cache的其他实现,基本都用于二级缓存;

所以本章后续的内容,都是针对PerpetualCache展开的。

执行器

一级缓存在BaseExecutor中就有体现,下面列出的是BaseExecutor构造函数。很明显,一级缓存是默认开启的,和cacheEnabled是否设置为true没有关系。

  protected BaseExecutor(Configuration configuration, Transaction transaction) {
    this.transaction = transaction;
    this.deferredLoads = new ConcurrentLinkedQueue<>();
    // 本地缓存,新建PerpetualCache对象
    this.localCache = new PerpetualCache("LocalCache");
    this.localOutputParameterCache = new PerpetualCache("LocalOutputParameterCache");
    this.closed = false;
    this.configuration = configuration;
    this.wrapper = this;
  }

执行过程

用户每一次与数据库交互,Mybatis都会创建一个SqlSession对象,在sqlSession中有本地缓存对象,在同一个会话中如果出现重复查询,都会根据查询条件去本地缓存中查询是否存在,如果有缓存,则从缓存中读取数据,直接返回给调用方。否则从数据库中读取数据,加载到缓存中,然后返回给用户。 水煮MyBatis(十二)- 一级缓存

大概分为五步:

  1. 根据查询条件,查询语句等条件,生成缓存key;
  2. 根据key去本地缓存中查找,如果找到则返回给用户;
  3. 如果没有缓存,则去数据库中查询;
  4. 从数据库中读取到数据之后,将数据放到缓存;
  5. 返回

生命周期

会话级本地缓存的生命周期基本上是和sqlSession绑定的,同生共死的关系,毕竟其是在创建sqlSession时,作为内部属性对象生成的。当然也有一些例外,比如在会话活动期间,穿插执行了更新或者删除语句,那么会删除对应缓存,避免返回脏数据,造成业务故障。 缓存删除的常见场景:

  1. 更新/删除DB数据;
  2. 在Mapper接口方法定义了更新策略:Options.FlushCachePolicy.TRUE;
  3. sqlSession关闭;

下面列出的是BaseExecutor里的update方法,在执行具体更新方法之前,就会强制清除缓存。

  public int update(MappedStatement ms, Object parameter) throws SQLException {
    // 清除缓存
    clearLocalCache();
    // 执行继承类里的方法
    return doUpdate(ms, parameter);
  }

更新策略

下面是Mapper里的一个查询方法,定义了更新策略为true,即每次执行方法都需要更新缓存

    @Select("select * from tb_image where md5 = #{md5}")
    @Options(flushCache = Options.FlushCachePolicy.TRUE, useCache = true)
    ImageInfo byMd5(@Param(value = "md5") String md5);

使用此配置的地方,在MapperAnnotationBuilder里构建方法MappedStatement对象的时候:

      boolean isSelect = sqlCommandType == SqlCommandType.SELECT;
      boolean flushCache = !isSelect;
      boolean useCache = isSelect;
      if (options != null) {
        if (FlushCachePolicy.TRUE.equals(options.flushCache())) {
          flushCache = true;
        } else if (FlushCachePolicy.FALSE.equals(options.flushCache())) {
          flushCache = false;
        }
        useCache = options.useCache();
      }

逻辑如下:

  • 默认情况下,非查询方法需要更新缓存,查询方法会用到缓存;
  • 如果配置了@Options注解,则根据更新策略,设置flushCache的值;
  • useCache也会根据@Options的配置进行重新赋值;

缓存结构

对于PerpetualCache来说,结构是比较简单的,缓存数据都保存在一个Map里面。id是在创建的时候指定的,比如上文提到的LocalCache、LocalOutputParameterCache。

public class PerpetualCache implements Cache {
  // 缓存名称,比如上文提到的LocalCache、LocalOutputParameterCache
  private final String id;
  // 缓存结构
  private final Map<Object, Object> cache = new HashMap<>();
  // 构造方法
  public PerpetualCache(String id) {
    this.id = id;
  }
  ...
}

Key的生成

  public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {
    
    CacheKey cacheKey = new CacheKey();
    // ms的id,一般是Mapper的类路径+方法名称
    cacheKey.update(ms.getId());
    // 分页参数,表坐标
    cacheKey.update(rowBounds.getOffset());
    // 分页参数,数据条数限制
    cacheKey.update(rowBounds.getLimit());
    // 在Mapper方法里定义的查询语句
    cacheKey.update(boundSql.getSql());
    List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
    // 读取参数
    for (ParameterMapping parameterMapping : parameterMappings) {
      // 提取输入参数
      if (parameterMapping.getMode() != ParameterMode.OUT) {
        String propertyName = parameterMapping.getProperty();
        // 读取输入参数的值
        cacheKey.update(getValue(propertyName));
      }
    }
    if (configuration.getEnvironment() != null) {
      // 环境id,内容为:SqlSessionFactoryBean.class.getSimpleName()
      cacheKey.update(configuration.getEnvironment().getId());
    }
    return cacheKey;
  }

主要由五个元素构成: 1.MappedStatement 的id,一般是Mapper的类路径+方法名称; 2.分页参数,表坐标offset; 3.分页参数,数据条数限制limit; 4.在Mapper方法里定义的查询语句; 5.输入参数的值; 6.环境id,内容为:SqlSessionFactoryBean.class.getSimpleName()

上文中提到的查询方法byMd5,生成的key为: 271108400:3910348202:com.essay.mybatis.mapper.ImageInfoMapper.byMd5:0:2147483647:select * from tb_image where md5 = ?:6e705a7733ac5gbwopmp02:SqlSessionFactoryBean

思考

如何才能用到一级缓存?

用到一级缓存的前提条件是在同一个sqlSession的生命周期之内,多次重复查询。如果用spring注入的mapper,在同一个事务之内进行多次查询,是不能用到一级缓存的,因为每个查询方法,都会创建一个executor来执行,执行完成之后关闭sqlSession,而一级缓存是绑定的executor的。

手动获取sqlSession

下面这个测试方法,用sqlSession手动获取mapper,然后在其未关闭之前进行两次查询,就会用到一级缓存了。

    @Test
    public void sqlSession() {
        SqlSession sqlSession = sqlSessionFactory.openSession();
        ImageInfoMapper mapper = sqlSession.getMapper(ImageInfoMapper.class);
        ImageInfo info = mapper.byMd5("6e705a7733ac5gbwopmp02");
        log.info(">>>>>>>>>>>>>>>>>>>=>,{}", info);
        info = mapper.byMd5("6e705a7733ac5gbwopmp02");
        log.info("=>,{}", info);
    }

事务内

在同一个事务里面,sqlSession也是复用的,所以也会用到一级缓存。如果把@Transactional注解拿掉,则在方法执行期间,会创建两个sqlSession,这种情况下,两个sqlSession的一级缓存是隔离开的,需要走两次数据库查询。

    @Transactional(rollbackFor = Exception.class)
    public void selectDuplicateTransaction(String md5){
        ImageInfo info = imageInfoMapper.byMd5(md5);
        log.info(">>>>>>>>>>>>>>>>>>>=>,{}", info);
        ImageInfo info1  = imageInfoMapper.byMd5(md5);
        log.info(">>>>>>>>>>>>>>>>>>>=>,{}", info1);
    }

内存溢出?

在一级缓存里,是没有自动过期的概念的,cache也没有容量限制,如果一个sqlSession生存时间足够长,是有可能导致内存溢出的。这就需要开发者注意了,一般情况下,使用spring注入的mapper对象进行查询,是不会有这样的疑虑的,因为sqlSession的生存周期非常短暂,也很少重复利用。

转载自:https://juejin.cn/post/7244409860833738789
评论
请登录