首页 > 其他分享 >mybatis缓存

mybatis缓存

时间:2023-08-11 16:44:06浏览次数:65  
标签:mapper 缓存 ms cache session key mybatis

一级缓存

一级缓存是同一session内缓存,随着session的关闭而被清除。

先看下效果

        String resource = "mybatis-config.xml";
        SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream(resource));
        SqlSession sqlSession = sessionFactory.openSession();
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        User user1 = mapper.findUserByNo("001");
        System.out.println(user1);
        User user2 = mapper.findUserByNo("001");
        System.out.println(user2);
        System.out.println(user1.equals(user2));

执行两次相同的mapper.findUserByNo方法,观察日志只向数据库发送一次查询请求。并且user1.equals(user2)是完全相同的两个对象。证明缓存命中,第二次查询读的缓存。

命中条件

必须同一会话session这个就不用多说了

必须是相同的mapper方法,相同的参数

mapper.findUserByNo("001");
mapper.findUserByNo("002");

这样不同的参数是不会命中缓存

中间没有执行过更新(update,insert,delete)操作

mapper.findUserByNo("001");
mapper.updateUser("002");
mapper.findUserByNo("001");

更新操作会清空缓存

session.clearCache()会清空缓存

源码实现

****CacheExecutor主要用来对Executor进行包装完成缓存的处理

cacheKey的创建

public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
  BoundSql boundSql = ms.getBoundSql(parameterObject);
  //创建cacheKey
  CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql);
  return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
}

创建cacheKey方法。CacheKey有一个update方法,加入新参数会重新构造计算该类的hashcode方法。

public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {
  if (closed) {
    throw new ExecutorException("Executor was closed.");
  }
  CacheKey cacheKey = new CacheKey();
  //语句id。就是mapper方法的全路径 包名+类名+方法名
  cacheKey.update(ms.getId());
  //分页条件
  cacheKey.update(rowBounds.getOffset());
  cacheKey.update(rowBounds.getLimit());
  //sql语句
  cacheKey.update(boundSql.getSql());
  //sql参数
  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因子
      cacheKey.update(value);
    }
  }
  if (configuration.getEnvironment() != null) {
    // 最后将运行环境值加
    cacheKey.update(configuration.getEnvironment().getId());
  }
  return cacheKey;
}

从key的构造可以看出调用必须是同一个mapper方法并且参数值相等。

缓存的存入和获取

在BaseExecutor.query方法中可以看到使用的PerpetualCache来存储缓存,其内部也是维护了一个map用来存储缓存数据。

BaseExecutor中操作缓存方法

public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
  ...
  List<E> list;
  try {
    queryStack++;
    //判断是否指定resultHandler,否则从本地缓存根据cacheKey获取数据
    list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;
    if (list != null) {//缓存命中
      handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);
    } else {//否则从数据库查询
      list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);
    }
  } finally {
    queryStack--;
  }
  ...
  return list;
}

  private <E> List<E> queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    List<E> list;
    localCache.putObject(key, EXECUTION_PLACEHOLDER);
    try {//具体执行数据库查询
      list = doQuery(ms, parameter, rowBounds, resultHandler, boundSql);
    } finally {
      localCache.removeObject(key);
    }
    //将查询结果存放到cache中
    localCache.putObject(key, list);
    if (ms.getStatementType() == StatementType.CALLABLE) {
      localOutputParameterCache.putObject(key, parameter);
    }
    return list;
  }
  
  public void clearLocalCache() {
    if (!closed) {//清空本地缓存
      localCache.clear();
      localOutputParameterCache.clear();
    }
  }

在query方法中会先根据cacheKey进行缓存查找,如果找不到在使用queryFromDatabase方法进行数据库查询,数据库查询完后会将结果加入缓存。另外在Executor的update方法(update、insert、delete最后都会交给update方法),session的clearCache方法最后都会调用clearLocalCache来清空缓存。所以上面说缓存失效的几种场景就很好理解了。

生命周期

来看下几个对象的关系

public class DefaultSqlSession implements SqlSession {
  // 这里会是一个CacheExecutor实例
  private final Executor executor {
  	  /**
  	   *CacheExecutor的delegate是一个SimpleExecutor,SimpleExecutor继承自BaseExecutor
  	   */
      Executor delegate {
        //缓存对象
        PerpetualCache localCache;
      }
  };
}  

从上面的对象关系可以看出,缓存是session对象一个属性。会随着session的关闭二消失。

使用场景

有什么用呢?

二级缓存

开启使用

上面说的一级缓存是在同一session会话中,很有局限性,当session关闭时候缓存就消失了。二级缓存是session间共享的。

在mapper里添加配置开启二级缓存

<mapper namespace="com.test.mapper.UserMapper" >
   <cache ></cache>
</mapper>

        String resource = "mybatis-config.xml";
        SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(Resources.getResourceAsStream(resource));
        //开启第一个session
        SqlSession sqlSession = sessionFactory.openSession();
        UserMapper mapper = sqlSession.getMapper(UserMapper.class);
        User user1 = mapper.findUserByNo("001");
        System.out.println(user1);
        sqlSession.close();//关闭第一个session
        //开启第二个session
        sqlSession = sessionFactory.openSession();
        mapper = sqlSession.getMapper(UserMapper.class);
        User user2 = mapper.findUserByNo("001");
        System.out.println(user2);
        System.out.println(user1.equals(user2));

上面的程序执行开启两个session,执行相同的mapper方法,参数也一致。然后会发现只会向数据库发出一次查询请求,第二次走的缓存。

一个查询首先会从二级缓存查找,然后在从一级缓存查找,最后走数据库查询。

缓存的存储

具体代码看CacheExecutor.query方法主要逻辑

public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)
    throws SQLException {
   //从statment获取缓存,这里就是二级缓存
  Cache cache = ms.getCache();
  if (cache != null) {
    flushCacheIfRequired(ms);
    if (ms.isUseCache() && resultHandler == null) {
      //从缓存中获取
      List<E> list = (List<E>) tcm.getObject(cache, key);
      if (list == null) {//未命中,走查询
        list = delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
        tcm.putObject(cache, key, list); // 将查询结果放入缓存
      }
      return list;
    }
  }
  return delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
}

这里有两个主要的变量tcm和cache。 cache是从MappedStatement获取的,而statement又是configuration初始化的时候创建的,因此是和mybatis同生命周期的,全局的。 这里cache实例的结构是这样的
image

cache是层层delegate包装。最后存储结构还是map。这些Cache类都实现了Cache接口。Cache三个主要接口方法就是putObject、getObject和removeObject。

SynchronizedCache是在缓存操作时候方法都加上了synchronized。

LoggingCache在缓存命中记录命中率。

SerializedCache对缓存数据对象都进行序列化和反序列化操作。这时候你就要知道为什么mapper返回的对象都要实现序列化接口了。

LruCache实现LRU(least recently used)算法。缓存过大时候移除策略。

PerpetualCache这个就是是包装了下map。

tcm变量是TransactionalCacheManager类实例。里面存有本地session所持有各statment的cache。

Map<Cache, TransactionalCache> transactionalCaches;

key是Cache类型,代表不同的二级缓存。value是TransactionalCache。这个也实现了Cache接口。当session提交或rollback时候,会将session内所持有的缓存(tcm中保存)依次进行提交到二级缓存或清空。

TransactionalCacheManager类的代码

public class TransactionalCacheManager {

  private final Map<Cache, TransactionalCache> transactionalCaches = new HashMap<>();
  //清空
  public void clear(Cache cache) {
    getTransactionalCache(cache).clear();
  }

  public Object getObject(Cache cache, CacheKey key) {
    return getTransactionalCache(cache).getObject(key);
  }
  
  public void putObject(Cache cache, CacheKey key, Object value) {
    getTransactionalCache(cache).putObject(key, value);
  }
  //提交
  public void commit() {
    for (TransactionalCache txCache : transactionalCaches.values()) {
      txCache.commit();
    }
  }
  //回滚清空
  public void rollback() {
    for (TransactionalCache txCache : transactionalCaches.values()) {
      txCache.rollback();
    }
  }
  //根据二级缓存对象获取当前session中是否有本地缓存
  private TransactionalCache getTransactionalCache(Cache cache) {
    //如果map没有就new一个TransactionalCache,然后cache会传入作为其delegate
    return MapUtil.computeIfAbsent(transactionalCaches, cache, TransactionalCache::new);
  }

}

TransactionalCache提交到二级缓存方法

private void flushPendingEntries() {
  for (Map.Entry<Object, Object> entry : entriesToAddOnCommit.entrySet()) {
    //这里的delegate是具体的二级缓存,transactionalCaches对应的key
    delegate.putObject(entry.getKey(), entry.getValue());
  }
  for (Object entry : entriesMissedInCache) {
    if (!entriesToAddOnCommit.containsKey(entry)) {
      delegate.putObject(entry, null);
    }
  }
}

这样二级缓存的存储和获取就都明白了。会过头来看下二级缓存是怎么初始化的呢。这个时候就要看下mybatis初始化过程,主要在解析mapper文件的时候

这里主要在XMLMapperBuilder类中进行处理。入口是parse方法,然后调用configurationElement方法,

private void configurationElement(XNode context) {
    String namespace = context.getStringAttribute("namespace");
    builderAssistant.setCurrentNamespace(namespace);
    cacheRefElement(context.evalNode("cache-ref"));
    //解析cache配置
    cacheElement(context.evalNode("cache"));
    parameterMapElement(context.evalNodes("/mapper/parameterMap"));
    resultMapElements(context.evalNodes("/mapper/resultMap"));
    sqlElement(context.evalNodes("/mapper/sql"));
    buildStatementFromContext(context.evalNodes("select|insert|update|delete"));
  
}
private void cacheElement(XNode context) {
    if (context != null) {//解析所有的cache节点配置内容
     //基础缓存类型,这里是PerpetualCache类
      String type = context.getStringAttribute("type", "PERPETUAL");
      Class<? extends Cache> typeClass = typeAliasRegistry.resolveAlias(type);
      //下面是所有的cache配置
      String eviction = context.getStringAttribute("eviction", "LRU");
      Class<? extends Cache> evictionClass = typeAliasRegistry.resolveAlias(eviction);
      Long flushInterval = context.getLongAttribute("flushInterval");
      Integer size = context.getIntAttribute("size");
      boolean readWrite = !context.getBooleanAttribute("readOnly", false);
      boolean blocking = context.getBooleanAttribute("blocking", false);
      Properties props = context.getChildrenAsProperties();
      //创建缓存实例
      builderAssistant.useNewCache(typeClass, evictionClass, flushInterval, size, readWrite, blocking, props);
    }
  }
  public Cache useNewCache(Class<? extends Cache> typeClass,
      Class<? extends Cache> evictionClass,
      Long flushInterval,
      Integer size,
      boolean readWrite,
      boolean blocking,
      Properties props) {
    Cache cache = new CacheBuilder(currentNamespace)
        .implementation(valueOrDefault(typeClass, PerpetualCache.class))
        .addDecorator(valueOrDefault(evictionClass, LruCache.class))
        .clearInterval(flushInterval)
        .size(size)
        .readWrite(readWrite)
        .blocking(blocking)
        .properties(props)
        .build();//将所有的配置属性设置好后,build创建实例
    configuration.addCache(cache);
    currentCache = cache;
    return cache;
  }

再来看CacheBuilder的build过程

public Cache build() {
  setDefaultImplementations();
  //第一步是基础实现PerpetualCache
  Cache cache = newBaseCacheInstance(implementation, id);
  setCacheProperties(cache);
  // issue #352, do not apply decorators to custom caches
  if (PerpetualCache.class.equals(cache.getClass())) {
    //这里所有的装饰cache,默认配置会有一个LruCache
    for (Class<? extends Cache> decorator : decorators) {
      cache = newCacheDecoratorInstance(decorator, cache);
      setCacheProperties(cache);
    }
    //下面对cache进行一些标准装饰
    cache = setStandardDecorators(cache);
  } else if (!LoggingCache.class.isAssignableFrom(cache.getClass())) {
    cache = new LoggingCache(cache);
  }
  return cache;
}
private Cache setStandardDecorators(Cache cache) {
      MetaObject metaCache = SystemMetaObject.forObject(cache);
      if (size != null && metaCache.hasSetter("size")) {
        metaCache.setValue("size", size);
      }
      if (clearInterval != null) {
        cache = new ScheduledCache(cache);
        ((ScheduledCache) cache).setClearInterval(clearInterval);
      }
      if (readWrite) {//cache配置readWrite为true,使用SerializedCache进行装饰
        cache = new SerializedCache(cache);
      }
      //添加log装饰
      cache = new LoggingCache(cache);
      //添加同步装饰
      cache = new SynchronizedCache(cache);
      if (blocking) {
        cache = new BlockingCache(cache);
      }
      return cache;
  }

看到这里就会明白为什么cache会包装这么多层了。我们看到LruCache和SerializedCache是可以通过配置去掉的。其它的几个装饰好像都是标准不能去掉的。

<cache
  eviction="LRU"
  flushInterval="60000"
  size="512"
  readOnly="false"/>

eviction配置对应缓存清理策略,默认是LRU会用到LruCache

readOnly对应是否是只读,默认false。可读写就会用到SerializedCache。

size缓存大小,默认1024。超过就会调用LruCache清理

flushInterval刷新间隔,默认是不设置,如果设置就会启用ScheduledCache。

缓存的清除

默认情况下select不会清空缓存,update,insert和delete都会清空statment的缓存。这是默认配置。可以在mapper文件的statment语句通过flushCache属性配置。

回到CacheExecutor类,flushCacheIfRequired用来清空缓存。query和update方法都会调用该方法。

 public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)
      throws SQLException {
    Cache cache = ms.getCache();
    if (cache != null) {
      flushCacheIfRequired(ms);
      ...
      }
    }
    return delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
  }
public int update(MappedStatement ms, Object parameterObject) throws SQLException {
  flushCacheIfRequired(ms);
  return delegate.update(ms, parameterObject);
}
private void flushCacheIfRequired(MappedStatement ms) {
  Cache cache = ms.getCache();
  //根据flushCache配置是否清空缓存
  if (cache != null && ms.isFlushCacheRequired()) {
    tcm.clear(cache);
  }
}

这里tcm的key是共享的二级缓存Cache实例引用,在session内清了,也相当于整体清理了。引用也有一定好处。一处改处处改。这也是为什么要使用SynchronizedCache包装的原因吧。

使用场景

一些公共配置,字典,菜单,权限机构等等使用二级缓存可以提高效率。

总结

一级缓存是在同一个session可见。对在同一个session内多次相同的查询生效。二级缓存是session间共享的。缓存首先会从二级缓存查找,然后是一级缓存。由缓存CacheKey的构建可以知道,必须是调用同一个mapper的相同方法并且实例参数一致才被判断为相同查询。二级缓存的存储是以MappedStatement为单位的,也就是一个select标签方法。同一个session可能同时持有多个二级缓存,二级缓存的更新是在session提交或close的时候将本地缓存更新到二级缓存。

标签:mapper,缓存,ms,cache,session,key,mybatis
From: https://www.cnblogs.com/bird2048/p/17623384.html

相关文章

  • Mybatis
    MybatisMyBatis是一款优秀的持久层框架,用于简化JDBC的开发。官网:https://mybatis.org/mybatis-3/zh/index.html入门使用Mybatis操作数据库,就是在Mybatis中编写SQL查询代码,发送给数据库执行,数据库执行后返回结果。Mybatis操作数据库的步骤:1.准备工作(创建springboot工程、......
  • Caffeine本地缓存
    参考:https://blog.csdn.net/yingyujianmo/article/details/122755222......
  • Mybatis-plus SQL效率插件PerformanceInterceptor无效->替换为p6spy
    使用mybatis-plus时,需要加入执行的sql分析发现mybatis-plus中的PerformanceInterceptor无效了查了信息发现3.2.0版本之后把这个功能可剔除了可同等替换为p6spy插件添加依赖第一<dependency><groupId>p6spy</groupId><artifactId>p6sp......
  • 资源过滤器—MVC中使用资源过滤器实现不执行Action方法体读取缓存信息返回
    前言上两篇文章分享了过滤器实现JWT进行鉴权,分别是通过授权过滤器和操作过滤器实现,这两个过滤器也是最常用的。文章链接:授权过滤器—MVC中使用授权过滤器实现JWT权限认证,操作过滤器—MVC中使用操作过滤器实现JWT权限认证,接下来将简单的谈谈资源过滤器在MVC中如何使用,一般项目中这......
  • Springboot 3.x 使用PageHelper实现MyBatis分页查询
    开发环境SpringBoot3.0.1Maven工程JDKOpenJdk17.0.6引入pom依赖<dependency><groupId>com.github.pagehelper</groupId><artifactId>pagehelper-spring-boot-starter</artifactId><version>1.4.7</version></depende......
  • mybatis源码阅读
    配置解析首先来看一个简单使用例子Stringresource="mybatis-config.xml";//读取配置,创建sessionFactorySqlSessionFactorysessionFactory=newSqlSessionFactoryBuilder().build(Resources.getResourceAsStream(resource));//opensessionSqlSessionsqlSession=sess......
  • MyBatis Plus 大数据量查询优化
    大数据量操作的场景大致如下:数据迁移数据导出批量处理数据在实际工作中当指定查询数据过大时,我们一般使用分页查询的方式一页一页的将数据放到内存处理。但有些情况不需要分页的方式查询数据或分很大一页查询数据时,如果一下子将数据全部加载出来到内存中,很可能会发生OOM(内存溢出);......
  • vue cli 解决浏览器缓存问题
    在vue打包时会遇到前端明明发布了,但是浏览器却没有更新。需要强制刷新才能看到最新的内容。解决方法一加时间戳后缀在vue.config.js的文件中加入constTimestamp=newDate().getTime();module.exports={configureWebpack:{//webpack配置output:{//输出重构......
  • Mybatis
    MybatisresultMap首先要了解,一个resultMap中都有些什么常用的玩意:展开代码<resultMapid="唯一标识"type="映射的entity对象的绝对路径"><idcolumn="表主键字段"jdbcType="字段类型"property="映射entity对象的主键属性"/><resultco......
  • MyBatis Generator 学习记录
    目录参考资料什么是MyBatisGenerator?运行MyBatisGenerator方式mavenplugin方式java代码方式参考资料官方文档什么是MyBatisGenerator?MyBatisGenerator是MyBatis代码生成工具。运行MyBatisGenerator方式命令行antmaven运行java代码运行eclipse......