官网文档

MyBatis 内置了一个弱小的事务性查问缓存机制,它能够十分不便地配置和定制。本篇文章,小编将会在最短的工夫呢,通过观察源码来粗浅理解Mybatis的
一级二级缓存;而后在说如何定制。

一、Mybatis Cache设计

在Mybatis中所有的缓存,都是实现自Cache接口。无论是一级缓存还是二级缓存都是实现这个接口。其中一级缓存是本地缓存,二级缓存是一个容许开发者扩大的
缓存(eg: ehcache/或者内置的很多缓存)。

public interface Cache {  String getId();  void putObject(Object key, Object value);  Object getObject(Object key);  Object removeObject(Object key);  void clear();  int getSize();  default ReadWriteLock getReadWriteLock() {    return null;  }}

二、一级缓存

一级缓存是本地缓存,其实就是PerpetualCache这类,它的源码也很简略,其实就是一个Map而已。个别面试的常常说一级缓存称为
SqlSession缓存,咱们看其实最终实现是在BaseExecutor进行做的。就这么简略。

public abstract class BaseExecutor implements Executor {    // 一级缓存本地缓存    protected PerpetualCache localCache;        protected BaseExecutor(Configuration configuration, Transaction transaction) {        this.transaction = transaction;        this.deferredLoads = new ConcurrentLinkedQueue<>();        this.localCache = new PerpetualCache("LocalCache");    }        // 执行查问后增加到一级缓存中    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);        }        localCache.putObject(key, list);        if (ms.getStatementType() == StatementType.CALLABLE) {          localOutputParameterCache.putObject(key, parameter);        }        return list;      }}  

三、二级缓存

二级缓存是基于装璜器模式,它容许开发者自定义缓存的实现,只有实现了Cache接口就行。通过装璜器的设计。
CachingExecutor从MappedStatement#getCache获取缓存的具体实现,从而进行缓存操作。

上面代码是看Mybatis是如何进行装璜器的。留神看正文。如果开启缓存,则包装器对Executor进行包装。

public class Configuration {    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);        }        // 如果开启缓存,则包装器对Executor进行包装        if (cacheEnabled) {          executor = new CachingExecutor(executor);        }        executor = (Executor) interceptorChain.pluginAll(executor);        return executor;  }}

CachingExecutor在理论执行时候从MappedStatement#getCache获取缓存的具体实现,从而进行缓存操作。
看到查问是先从二级缓存中获取,如果没有获取到就从一级缓存中获取,还没有就查问db。

public class CachingExecutor implements Executor {  @Override  public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)      throws SQLException {    // 从MappedStatement获取Cache    Cache cache = ms.getCache();    if (cache != null) {      flushCacheIfRequired(ms);      if (ms.isUseCache() && resultHandler == null) {        ensureNoOutParams(ms, boundSql);        @SuppressWarnings("unchecked")        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); // issue #578 and #116        }        return list;      }    }    return delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);  }}

留神这里能够看到如果指定了要进行缓存,然而没有指定缓存的type默认是 PERPETUAL(PerpetualCache
)

四、开启二级缓存

4.1 内置二级缓存

  1. 首先开启配置
  2. 同时在Mapper文件中增加<cache/>标签 (XMLMapperBuilder#cacheElement)
  3. 或者是在Mapper类上增加@CacheNamespace注解(MapperAnnotationBuilder#parseCache)
<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE configuration        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"        "http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration>    <!-- 指定Mybatis应用log4j -->    <settings>        <setting name="logImpl" value="LOG4J"/>        // 通过 cacheEnabled 进行配置,如果不配置默认是true        <setting name="cacheEnabled" value="false"/>    </settings></configuration><?xml version="1.0" encoding="UTF-8"?><!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"><mapper namespace="orm.example.dal.mapper.TUserMapper">    // 增加cache标签    <cache/></mapper>    
属性含意
eviction缓存回收策略
flushInterval缓存刷新距离,缓存多长时间刷新一次,默认不清空,设置一个毫秒值
readOnly是否只读;true 只读
size缓存寄存多少个元素
type指定自定义缓存的全类名(实现Cache 接口即可)
blocking若缓存中找不到对应的key,是否会始终blocking,直到有对应的数据进入缓存。

一共能够应用的二级缓存有以下这些。

4.2 外置二级缓存

只有实现了Cache接口那么Mybatis就会调用这个接口实现进行缓存。上面只说一个思路。如下通过指定EhcacheCache
就能够将这个二级缓存的能力,交给Mybatis进行调用了。

<cache type="org.mybatis.caches.ehcache.EhcacheCache" >       <property name="timeToIdleSeconds" value="3600"/>      <property name="timeToLiveSeconds" value="3600"/>      <!-- 同ehcache参数maxElementsInMemory -->    <property name="maxEntriesLocalHeap" value="1000"/>    <!-- 同ehcache参数maxElementsOnDisk -->      <property name="maxEntriesLocalDisk" value="10000000"/>      <property name="memoryStoreEvictionPolicy" value="LRU"/></cache>

感谢您的浏览,本文由 西魏陶渊明 版权所有。如若转载,请注明出处:西魏陶渊明(https://blog.springlearn.cn/)

本文由mdnice多平台公布