QCon 演讲火热征集中,快来分享技术实践与洞见! 了解详情
写点什么

聊聊 MyBatis 缓存机制

  • 2020-02-27
  • 本文字数:11513 字

    阅读完需:约 38 分钟

聊聊MyBatis缓存机制

前言

MyBatis 是常见的 Java 数据库访问层框架。在日常工作中,开发人员多数情况下是使用 MyBatis 的默认缓存配置,但是 MyBatis 缓存机制有一些不足之处,在使用中容易引起脏数据,形成一些潜在的隐患。个人在业务开发中也处理过一些由于 MyBatis 缓存引发的开发问题,带着个人的兴趣,希望从应用及源码的角度为读者梳理 MyBatis 缓存机制。


本次分析中涉及到的代码和数据库表均放在 GitHub 上,地址: mybatis-cache-demo

目录

本文按照以下顺序展开。


  • 一级缓存介绍及相关配置。

  • 一级缓存工作流程及源码分析。

  • 一级缓存总结。

  • 二级缓存介绍及相关配置。

  • 二级缓存源码分析。

  • 二级缓存总结。

  • 全文总结。

一级缓存

一级缓存介绍

在应用运行过程中,我们有可能在一次数据库会话中,执行多次查询条件完全相同的 SQL,MyBatis 提供了一级缓存的方案优化这部分场景,如果是相同的 SQL 语句,会优先命中一级缓存,避免直接对数据库进行查询,提高性能。具体执行过程如下图所示。



每个 SqlSession 中持有了 Executor,每个 Executor 中有一个 LocalCache。当用户发起查询时,MyBatis 根据当前执行的语句生成MappedStatement,在 Local Cache 进行查询,如果缓存命中的话,直接返回结果给用户,如果缓存没有命中的话,查询数据库,结果写入Local Cache,最后返回结果给用户。具体实现类的类关系图如下图所示。


一级缓存配置

我们来看看如何使用 MyBatis 一级缓存。开发者只需在 MyBatis 的配置文件中,添加如下语句,就可以使用一级缓存。共有两个选项,SESSION或者STATEMENT,默认是SESSION级别,即在一个 MyBatis 会话中执行的所有语句,都会共享这一个缓存。一种是STATEMENT级别,可以理解为缓存只对当前执行的这一个Statement有效。


<setting name="localCacheScope" value="SESSION"/>
复制代码

一级缓存实验

接下来通过实验,了解 MyBatis 一级缓存的效果,每个单元测试后都请恢复被修改的数据。


首先是创建示例表 student,创建对应的 POJO 类和增改的方法,具体可以在 entity 包和 mapper 包中查看。


CREATE TABLE `student` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(200) COLLATE utf8_bin DEFAULT NULL, `age` tinyint(3) unsigned DEFAULT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
复制代码


在以下实验中,id 为 1 的学生名称是凯伦。

实验 1

开启一级缓存,范围为会话级别,调用三次getStudentById,代码如下所示:


public void getStudentById() throws Exception {    SqlSession sqlSession = factory.openSession(true); // 自动提交事务    StudentMapper studentMapper = sqlSession.getMapper(StudentMapper.class);    System.out.println(studentMapper.getStudentById(1));    System.out.println(studentMapper.getStudentById(1));    System.out.println(studentMapper.getStudentById(1));  }
复制代码


执行结果:



我们可以看到,只有第一次真正查询了数据库,后续的查询使用了一级缓存。

实验 2

增加了对数据库的修改操作,验证在一次数据库会话中,如果对数据库发生了修改操作,一级缓存是否会失效。


@Testpublic void addStudent() throws Exception {    SqlSession sqlSession = factory.openSession(true); // 自动提交事务    StudentMapper studentMapper = sqlSession.getMapper(StudentMapper.class);    System.out.println(studentMapper.getStudentById(1));    System.out.println("增加了" + studentMapper.addStudent(buildStudent()) + "个学生");    System.out.println(studentMapper.getStudentById(1));    sqlSession.close();}
复制代码


执行结果:



我们可以看到,在修改操作后执行的相同查询,查询了数据库,一级缓存失效

实验 3

开启两个SqlSession,在sqlSession1中查询数据,使一级缓存生效,在sqlSession2中更新数据库,验证一级缓存只在数据库会话内部共享。


@Testpublic void testLocalCacheScope() throws Exception {    SqlSession sqlSession1 = factory.openSession(true);    SqlSession sqlSession2 = factory.openSession(true);
StudentMapper studentMapper = sqlSession1.getMapper(StudentMapper.class); StudentMapper studentMapper2 = sqlSession2.getMapper(StudentMapper.class);
System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); System.out.println("studentMapper2更新了" + studentMapper2.updateStudentName("小岑",1) + "个学生的数据"); System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentById(1));}
复制代码



sqlSession2更新了 id 为 1 的学生的姓名,从凯伦改为了小岑,但 session1 之后的查询中,id 为 1 的学生的名字还是凯伦,出现了脏数据,也证明了之前的设想,一级缓存只在数据库会话内部共享。

一级缓存工作流程 &源码分析

那么,一级缓存的工作流程是怎样的呢?我们从源码层面来学习一下。

工作流程

一级缓存执行的时序图,如下图所示。


源码分析

接下来将对 MyBatis 查询相关的核心类和一级缓存的源码进行走读。这对后面学习二级缓存也有帮助。


SqlSession: 对外提供了用户和数据库之间交互需要的所有方法,隐藏了底层的细节。默认实现类是DefaultSqlSession



ExecutorSqlSession向用户提供操作数据库的方法,但和数据库操作有关的职责都会委托给 Executor。



如下图所示,Executor 有若干个实现类,为 Executor 赋予了不同的能力,大家可以根据类名,自行学习每个类的基本作用。



在一级缓存的源码分析中,主要学习BaseExecutor的内部实现。


BaseExecutorBaseExecutor是一个实现了 Executor 接口的抽象类,定义若干抽象方法,在执行的时候,把具体的操作委托给子类进行执行。


protected abstract int doUpdate(MappedStatement ms, Object parameter) throws SQLException;protected abstract List<BatchResult> doFlushStatements(boolean isRollback) throws SQLException;protected abstract <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException;protected abstract <E> Cursor<E> doQueryCursor(MappedStatement ms, Object parameter, RowBounds rowBounds, BoundSql boundSql) throws SQLException;
复制代码


在一级缓存的介绍中提到对Local Cache的查询和写入是在Executor内部完成的。在阅读BaseExecutor的代码后发现Local CacheBaseExecutor内部的一个成员变量,如下代码所示。


public abstract class BaseExecutor implements Executor {protected ConcurrentLinkedQueue<DeferredLoad> deferredLoads;protected PerpetualCache localCache;
复制代码


Cache: MyBatis 中的 Cache 接口,提供了和缓存相关的最基本的操作,如下图所示:



有若干个实现类,使用装饰器模式互相组装,提供丰富的操控缓存的能力,部分实现类如下图所示:



BaseExecutor成员变量之一的PerpetualCache,是对 Cache 接口最基本的实现,其实现非常简单,内部持有 HashMap,对一级缓存的操作实则是对 HashMap 的操作。如下代码所示:


public class PerpetualCache implements Cache { private String id; private Map<Object, Object> cache = new HashMap<Object, Object>();
复制代码


在阅读相关核心类代码后,从源代码层面对一级缓存工作中涉及到的相关代码,出于篇幅的考虑,对源码做适当删减,读者朋友可以结合本文,后续进行更详细的学习。


为执行和数据库的交互,首先需要初始化SqlSession,通过DefaultSqlSessionFactory开启SqlSession


private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {  ............  final Executor executor = configuration.newExecutor(tx, execType);  return new DefaultSqlSession(configuration, executor, autoCommit);}
复制代码


在初始化SqlSesion时,会使用Configuration类创建一个全新的Executor,作为DefaultSqlSession构造函数的参数,创建 Executor 代码如下所示:


public Executor newExecutor(Transaction transaction, ExecutorType executorType) {  executorType = executorType == null ? defaultExecutorType : executorType;  executorType = executorType == null ? ExecutorType.SIMPLE : executorType;  Executor executor;  if (ExecutorType.BATCH == executorType) {   executor = new BatchExecutor(this, transaction);  } else if (ExecutorType.REUSE == executorType) {   executor = new ReuseExecutor(this, transaction);  } else {   executor = new SimpleExecutor(this, transaction);  }  // 尤其可以注意这里,如果二级缓存开关开启的话,是使用CahingExecutor装饰BaseExecutor的子类  if (cacheEnabled) {   executor = new CachingExecutor(executor);  }  executor = (Executor) interceptorChain.pluginAll(executor);  return executor;}
复制代码


SqlSession创建完毕后,根据 Statment 的不同类型,会进入SqlSession的不同方法中,如果是Select语句的话,最后会执行到SqlSessionselectList,代码如下所示:


@Overridepublic <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {   MappedStatement ms = configuration.getMappedStatement(statement);   return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);}
复制代码


SqlSession把具体的查询职责委托给了 Executor。如果只开启了一级缓存的话,首先会进入BaseExecutorquery方法。代码如下所示:


@Overridepublic <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {  BoundSql boundSql = ms.getBoundSql(parameter);  CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);  return query(ms, parameter, rowBounds, resultHandler, key, boundSql);}
复制代码


在上述代码中,会先根据传入的参数生成 CacheKey,进入该方法查看 CacheKey 是如何生成的,代码如下所示:


CacheKey cacheKey = new CacheKey();cacheKey.update(ms.getId());cacheKey.update(rowBounds.getOffset());cacheKey.update(rowBounds.getLimit());cacheKey.update(boundSql.getSql());//后面是update了sql中带的参数cacheKey.update(value);
复制代码


在上述的代码中,将MappedStatement的 Id、SQL 的 offset、SQL 的 limit、SQL 本身以及 SQL 中的参数传入了 CacheKey 这个类,最终构成 CacheKey。以下是这个类的内部结构:


private static final int DEFAULT_MULTIPLYER = 37;private static final int DEFAULT_HASHCODE = 17;
private int multiplier;private int hashcode;private long checksum;private int count;private List<Object> updateList;
public CacheKey() { this.hashcode = DEFAULT_HASHCODE; this.multiplier = DEFAULT_MULTIPLYER; this.count = 0; this.updateList = new ArrayList<Object>();}
复制代码


首先是成员变量和构造函数,有一个初始的hachcode和乘数,同时维护了一个内部的updatelist。在CacheKeyupdate方法中,会进行一个hashcodechecksum的计算,同时把传入的参数添加进updatelist中。如下代码所示:


public void update(Object object) {  int baseHashCode = object == null ? 1 : ArrayUtil.hashCode(object);  count++;  checksum += baseHashCode;  baseHashCode *= count;  hashcode = multiplier* hashcode + baseHashCode;
updateList.add(object);}
复制代码


同时重写了CacheKeyequals方法,代码如下所示:


@Overridepublic boolean equals(Object object) {  .............  for (int i = 0; i < updateList.size(); i++) {   Object thisObject = updateList.get(i);   Object thatObject = cacheKey.updateList.get(i);   if (!ArrayUtil.equals(thisObject, thatObject)) {    return false;   }  }  return true;}
复制代码


除去 hashcode、checksum 和 count 的比较外,只要 updatelist 中的元素一一对应相等,那么就可以认为是 CacheKey 相等。只要两条 SQL 的下列五个值相同,即可以认为是相同的 SQL。


Statement Id + Offset + Limmit + Sql + Params


BaseExecutor 的 query 方法继续往下走,代码如下所示:


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);}
复制代码


如果查不到的话,就从数据库查,在queryFromDatabase中,会对localcache进行写入。


query方法执行的最后,会判断一级缓存级别是否是STATEMENT级别,如果是的话,就清空缓存,这也就是STATEMENT级别的一级缓存无法共享localCache的原因。代码如下所示:


if (configuration.getLocalCacheScope() == LocalCacheScope.STATEMENT) {    clearLocalCache();}
复制代码


在源码分析的最后,我们确认一下,如果是insert/delete/update方法,缓存就会刷新的原因。


SqlSessioninsert方法和delete方法,都会统一走update的流程,代码如下所示:


@Overridepublic int insert(String statement, Object parameter) {  return update(statement, parameter); }  @Override public int delete(String statement) {  return update(statement, null);}
复制代码


update方法也是委托给了Executor执行。BaseExecutor的执行方法如下所示:


@Overridepublic int update(MappedStatement ms, Object parameter) throws SQLException {  ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());  if (closed) {   throw new ExecutorException("Executor was closed.");  }  clearLocalCache();  return doUpdate(ms, parameter);}
复制代码


每次执行update前都会清空localCache


至此,一级缓存的工作流程讲解以及源码分析完毕。

总结

  1. MyBatis 一级缓存的生命周期和 SqlSession 一致。

  2. MyBatis 一级缓存内部设计简单,只是一个没有容量限定的 HashMap,在缓存的功能性上有所欠缺。

  3. MyBatis 的一级缓存最大范围是 SqlSession 内部,有多个 SqlSession 或者分布式的环境下,数据库写操作会引起脏数据,建议设定缓存级别为 Statement。

二级缓存

二级缓存介绍

在上文中提到的一级缓存中,其最大的共享范围就是一个 SqlSession 内部,如果多个 SqlSession 之间需要共享缓存,则需要使用到二级缓存。开启二级缓存后,会使用 CachingExecutor 装饰 Executor,进入一级缓存的查询流程前,先在 CachingExecutor 进行二级缓存的查询,具体的工作流程如下所示。



二级缓存开启后,同一个 namespace 下的所有操作语句,都影响着同一个 Cache,即二级缓存被多个 SqlSession 共享,是一个全局的变量。


当开启缓存后,数据的查询执行的流程就是 二级缓存 -> 一级缓存 -> 数据库。

二级缓存配置

要正确的使用二级缓存,需完成如下配置的。


  1. 在 MyBatis 的配置文件中开启二级缓存。


<setting name="cacheEnabled" value="true"/>
复制代码


  1. 在 MyBatis 的映射 XML 中配置 cache 或者 cache-ref 。


cache 标签用于声明这个 namespace 使用二级缓存,并且可以自定义配置。


<cache/>
复制代码


  • type:cache 使用的类型,默认是PerpetualCache,这在一级缓存中提到过。

  • eviction: 定义回收的策略,常见的有 FIFO,LRU。

  • flushInterval: 配置一定时间自动刷新缓存,单位是毫秒。

  • size: 最多缓存对象的个数。

  • readOnly: 是否只读,若配置可读写,则需要对应的实体类能够序列化。

  • blocking: 若缓存中找不到对应的 key,是否会一直 blocking,直到有对应的数据进入缓存。


cache-ref代表引用别的命名空间的 Cache 配置,两个命名空间的操作使用的是同一个 Cache。


<cache-ref namespace="mapper.StudentMapper"/>
复制代码

二级缓存实验

接下来我们通过实验,了解 MyBatis 二级缓存在使用上的一些特点。


在本实验中,id 为 1 的学生名称初始化为点点。

实验 1

测试二级缓存效果,不提交事务,sqlSession1查询完数据后,sqlSession2相同的查询是否会从缓存中获取数据。


@Testpublic void testCacheWithoutCommitOrClose() throws Exception {    SqlSession sqlSession1 = factory.openSession(true);    SqlSession sqlSession2 = factory.openSession(true);
StudentMapper studentMapper = sqlSession1.getMapper(StudentMapper.class); StudentMapper studentMapper2 = sqlSession2.getMapper(StudentMapper.class);
System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentById(1));}
复制代码


执行结果:



我们可以看到,当sqlsession没有调用commit()方法时,二级缓存并没有起到作用。

实验 2

测试二级缓存效果,当提交事务时,sqlSession1查询完数据后,sqlSession2相同的查询是否会从缓存中获取数据。


@Testpublic void testCacheWithCommitOrClose() throws Exception {    SqlSession sqlSession1 = factory.openSession(true);    SqlSession sqlSession2 = factory.openSession(true);
StudentMapper studentMapper = sqlSession1.getMapper(StudentMapper.class); StudentMapper studentMapper2 = sqlSession2.getMapper(StudentMapper.class);
System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); sqlSession1.commit(); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentById(1));}
复制代码



从图上可知,sqlsession2的查询,使用了缓存,缓存的命中率是 0.5。

实验 3

测试update操作是否会刷新该namespace下的二级缓存。


@Testpublic void testCacheWithUpdate() throws Exception {    SqlSession sqlSession1 = factory.openSession(true);    SqlSession sqlSession2 = factory.openSession(true);    SqlSession sqlSession3 = factory.openSession(true);
StudentMapper studentMapper = sqlSession1.getMapper(StudentMapper.class); StudentMapper studentMapper2 = sqlSession2.getMapper(StudentMapper.class); StudentMapper studentMapper3 = sqlSession3.getMapper(StudentMapper.class);
System.out.println("studentMapper读取数据: " + studentMapper.getStudentById(1)); sqlSession1.commit(); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentById(1));
studentMapper3.updateStudentName("方方",1); sqlSession3.commit(); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentById(1));}
复制代码



我们可以看到,在sqlSession3更新数据库,并提交事务后,sqlsession2StudentMapper namespace下的查询走了数据库,没有走 Cache。

实验 4

验证 MyBatis 的二级缓存不适应用于映射文件中存在多表查询的情况。


通常我们会为每个单表创建单独的映射文件,由于 MyBatis 的二级缓存是基于namespace的,多表查询语句所在的namspace无法感应到其他namespace中的语句对多表查询中涉及的表进行的修改,引发脏数据问题。


@Testpublic void testCacheWithDiffererntNamespace() throws Exception {    SqlSession sqlSession1 = factory.openSession(true);    SqlSession sqlSession2 = factory.openSession(true);    SqlSession sqlSession3 = factory.openSession(true);
StudentMapper studentMapper = sqlSession1.getMapper(StudentMapper.class); StudentMapper studentMapper2 = sqlSession2.getMapper(StudentMapper.class); ClassMapper classMapper = sqlSession3.getMapper(ClassMapper.class);
System.out.println("studentMapper读取数据: " + studentMapper.getStudentByIdWithClassInfo(1)); sqlSession1.close(); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentByIdWithClassInfo(1));
classMapper.updateClassName("特色一班",1); sqlSession3.commit(); System.out.println("studentMapper2读取数据: " + studentMapper2.getStudentByIdWithClassInfo(1));}
复制代码


执行结果:



在这个实验中,我们引入了两张新的表,一张 class,一张 classroom。class 中保存了班级的 id 和班级名,classroom 中保存了班级 id 和学生 id。我们在StudentMapper中增加了一个查询方法getStudentByIdWithClassInfo,用于查询学生所在的班级,涉及到多表查询。在ClassMapper中添加了updateClassName,根据班级 id 更新班级名的操作。


sqlsession1studentmapper查询数据后,二级缓存生效。保存在 StudentMapper 的 namespace 下的 cache 中。当sqlSession3classMapperupdateClassName方法对 class 表进行更新时,updateClassName不属于StudentMappernamespace,所以StudentMapper下的 cache 没有感应到变化,没有刷新缓存。当StudentMapper中同样的查询再次发起时,从缓存中读取了脏数据。

实验 5

为了解决实验 4 的问题呢,可以使用 Cache ref,让ClassMapper引用StudenMapper命名空间,这样两个映射文件对应的 SQL 操作都使用的是同一块缓存了。


执行结果:



不过这样做的后果是,缓存的粒度变粗了,多个Mapper namespace下的所有操作都会对缓存使用造成影响。

二级缓存源码分析

MyBatis 二级缓存的工作流程和前文提到的一级缓存类似,只是在一级缓存处理前,用CachingExecutor装饰了BaseExecutor的子类,在委托具体职责给delegate之前,实现了二级缓存的查询和写入功能,具体类关系图如下图所示。


源码分析

源码分析从CachingExecutorquery方法展开,源代码走读过程中涉及到的知识点较多,不能一一详细讲解,读者朋友可以自行查询相关资料来学习。


CachingExecutorquery方法,首先会从MappedStatement中获得在配置初始化时赋予的 Cache。


Cache cache = ms.getCache();
复制代码


本质上是装饰器模式的使用,具体的装饰链是:


SynchronizedCache -> LoggingCache -> SerializedCache -> LruCache -> PerpetualCache。



以下是具体这些 Cache 实现类的介绍,他们的组合为 Cache 赋予了不同的能力。


  • SynchronizedCache:同步 Cache,实现比较简单,直接使用 synchronized 修饰方法。

  • LoggingCache:日志功能,装饰类,用于记录缓存的命中率,如果开启了 DEBUG 模式,则会输出命中率日志。

  • SerializedCache:序列化功能,将值序列化后存到缓存中。该功能用于缓存返回一份实例的 Copy,用于保存线程安全。

  • LruCache:采用了 Lru 算法的 Cache 实现,移除最近最少使用的 Key/Value。

  • PerpetualCache: 作为为最基础的缓存类,底层实现比较简单,直接使用了 HashMap。


然后是判断是否需要刷新缓存,代码如下所示:


flushCacheIfRequired(ms);
复制代码


在默认的设置中SELECT语句不会刷新缓存,insert/update/delte会刷新缓存。进入该方法。代码如下所示:


private void flushCacheIfRequired(MappedStatement ms) {  Cache cache = ms.getCache();  if (cache != null && ms.isFlushCacheRequired()) {   tcm.clear(cache);  }}
复制代码


MyBatis 的CachingExecutor持有了TransactionalCacheManager,即上述代码中的 tcm。


TransactionalCacheManager中持有了一个 Map,代码如下所示:


private Map<Cache, TransactionalCache> transactionalCaches = new HashMap<Cache, TransactionalCache>();
复制代码


这个 Map 保存了 Cache 和用TransactionalCache包装后的 Cache 的映射关系。


TransactionalCache实现了 Cache 接口,CachingExecutor会默认使用他包装初始生成的 Cache,作用是如果事务提交,对缓存的操作才会生效,如果事务回滚或者不提交事务,则不对缓存产生影响。


TransactionalCache的 clear,有以下两句。清空了需要在提交时加入缓存的列表,同时设定提交时清空缓存,代码如下所示:


@Overridepublic void clear() {        clearOnCommit = true;        entriesToAddOnCommit.clear();}
复制代码


CachingExecutor继续往下走,ensureNoOutParams主要是用来处理存储过程的,暂时不用考虑。


if (ms.isUseCache() && resultHandler == null) {        ensureNoOutParams(ms, parameterObject, boundSql);
复制代码


之后会尝试从 tcm 中获取缓存的列表。


List<E> list = (List<E>) tcm.getObject(cache, key);
复制代码


getObject方法中,会把获取值的职责一路传递,最终到PerpetualCache。如果没有查到,会把 key 加入 Miss 集合,这个主要是为了统计命中率。


Object object = delegate.getObject(key);if (object == null) {        entriesMissedInCache.add(key);}
复制代码


CachingExecutor继续往下走,如果查询到数据,则调用tcm.putObject方法,往缓存中放入值。


if (list == null) {        list = delegate.<E> query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);        tcm.putObject(cache, key, list); // issue #578 and #116}
复制代码


tcm 的put方法也不是直接操作缓存,只是在把这次的数据和 key 放入待提交的 Map 中。


@Overridepublic void putObject(Object key, Object object) {  entriesToAddOnCommit.put(key, object);}
复制代码


从以上的代码分析中,我们可以明白,如果不调用commit方法的话,由于TranscationalCache的作用,并不会对二级缓存造成直接的影响。因此我们看看Sqlsessioncommit方法中做了什么。代码如下所示:


@Overridepublic void commit(boolean force) {  try {   executor.commit(isCommitOrRollbackRequired(force));
复制代码


因为我们使用了 CachingExecutor,首先会进入 CachingExecutor 实现的 commit 方法。


@Overridepublic void commit(boolean required) throws SQLException {  delegate.commit(required);  tcm.commit();}
复制代码


会把具体 commit 的职责委托给包装的Executor。主要是看下tcm.commit(),tcm 最终又会调用到TrancationalCache


public void commit() {  if (clearOnCommit) {   delegate.clear();  }  flushPendingEntries();  reset();}
复制代码


看到这里的clearOnCommit就想起刚才TrancationalCacheclear方法设置的标志位,真正的清理 Cache 是放到这里来进行的。具体清理的职责委托给了包装的 Cache 类。之后进入flushPendingEntries方法。代码如下所示:


private void flushPendingEntries() {  for (Map.Entry<Object, Object> entry : entriesToAddOnCommit.entrySet()) {   delegate.putObject(entry.getKey(), entry.getValue());  }  ................}
复制代码


flushPendingEntries 中,将待提交的 Map 进行循环处理,委托给包装的 Cache 类,进行putObject的操作。


后续的查询操作会重复执行这套流程。如果是insert|update|delete的话,会统一进入CachingExecutorupdate方法,其中调用了这个函数,代码如下所示:


private void flushCacheIfRequired(MappedStatement ms)
复制代码


在二级缓存执行流程后就会进入一级缓存的执行流程,因此不再赘述。

总结

  1. MyBatis 的二级缓存相对于一级缓存来说,实现了SqlSession之间缓存数据的共享,同时粒度更加的细,能够到namespace级别,通过 Cache 接口实现类不同的组合,对 Cache 的可控性也更强。

  2. MyBatis 在多表查询时,极大可能会出现脏数据,有设计上的缺陷,安全使用二级缓存的条件比较苛刻。

  3. 在分布式环境下,由于默认的 MyBatis Cache 实现都是基于本地的,分布式环境下必然会出现读取到脏数据,需要使用集中式缓存将 MyBatis 的 Cache 接口实现,有一定的开发成本,直接使用 Redis、Memcached 等分布式缓存可能成本更低,安全性也更高。

全文总结

本文对介绍了 MyBatis 一二级缓存的基本概念,并从应用及源码的角度对 MyBatis 的缓存机制进行了分析。最后对 MyBatis 缓存机制做了一定的总结,个人建议 MyBatis 缓存特性在生产环境中进行关闭,单纯作为一个 ORM 框架使用可能更为合适。

作者简介

  • 凯伦,美团点评后端研发工程师,2016 年毕业于上海海事大学,现从事美团点评餐饮平台相关的开发工作。


2020-02-27 10:491551

评论

发布
暂无评论
发现更多内容

京东T7团队技术4面:线程池+索引+Spring +分布式锁

Java 程序员 后端

全面解读!构建边云一体的智能应用技术实践

百度开发者中心

最佳实践 方法论 边缘计算 前沿科技

五、redis配置信息以及常用命令

Java 程序员 后端

五分钟带你了解Seata分布式事务

Java 程序员 后端

CSS页面设计稿构思与实现(三)之UI组件

Augus

CSS 11月日更

云原生时代下,容器安全的“四个挑战”和“两个关键

Java 程序员 后端

京东T7架构师手写的10万字Spring Boot详细学习笔记+源码免费下载

Java 程序员 后端

为了加快速度,Redis都做了哪些“变态”设计

Java 程序员 后端

主流的消息队列MQ比较,详解MQ的4类应用场景

Java 程序员 后端

云原生新边界——阿里云边缘计算云原生落地实践

Java 程序员 后端

Serverless 下的微服务实践

Serverless Devs

阿里云 Serverless 架构 微服务

模块三

侠客行

架构实战 「架构实战营」

为什么你不应该恨Java!

Java 程序员 后端

为什么你的insert就死锁了

Java 程序员 后端

这一次,中国CRM站到风口上了吗?

ToB行业头条

CRM

京东T7架构师用470页就把微服务架构原理与开发实战文档讲完了(1)

Java 程序员 后端

为什么建议大家使用 Linux 开发?爽(外加七个感叹号)

Java 程序员 后端

事大发了!小助理告诉我:小伙伴21天斩获字节offer的关键竟是这份面试题!

Java 程序员 后端

五分钟带你读懂!Spring Cloud构建微服务分布式云平台(1)

Java 程序员 后端

五位阿里大牛联手撰写的《深入浅出Java多线程》,站在巨人的肩膀上学习!

Java 程序员 后端

2021 大厂面试题之 mysql 篇

小麦

MySQL 面试题 大厂

小学妹与我畅聊黑客渗透技术

喀拉峻

黑客 网络安全 信息安全 渗透测试

五分钟学会做一个在线抽奖系统,手把手教你抽奖还学不会嘛?

Java 程序员 后端

京东T7架构师用470页就把微服务架构原理与开发实战文档讲完了

Java 程序员 后端

五分钟带你读懂!Spring Cloud构建微服务分布式云平台

Java 程序员 后端

docker基础

小麦

二维码扫码登录是什么原理?

Java 程序员 后端

三年沉淀,Apache ShardingSphere 5.0.0 开启数据应用新篇章

SphereEx

Java 数据库 开源 Apache ShardingSphere DateBase

Serverless 工程实践 | 快速搭建 Kubeless 平台

Serverless Devs

k8s 架构设计 框架搭建 框架学习

Serverless 架构模式及演进

Serverless Devs

Serverless 架构设计

京东员工被裁后哀叹:优化后连面试的机会都没有,每月还要还贷款

Java 程序员 后端

聊聊MyBatis缓存机制_文化 & 方法_美团技术团队_InfoQ精选文章