Mybatis执行器BatchExecutor、ReuseExecutor、SimpleExecutor介绍

一、关系

这里说下Executor接口和他的子类之间的关系

//最顶层的接口
public interface Executor {}
//接着是基础BaseExecutor
public abstract class BaseExecutor implements Executor {}

BaseExecutor有三个实现类BatchExecutor、ReuseExecutor、SimpleExecutor是并列关系。

public class BatchExecutor extends BaseExecutor {}//批量执行器BatchExecutor
public class ReuseExecutor extends BaseExecutor {}//复用执行器ReuseExecutor
public class SimpleExecutor extends BaseExecutor {}//简单执行器SimpleExecutor

2、CachingExecutor

CachingExecutor是对执行器和缓存的一个封装,Caching [ko fin]

public class CachingExecutor implements Executor {
	//Executor的实现类对象,可以是BatchExecutor、ReuseExecutor、SimpleExecutor
    private final Executor delegate;
    private final TransactionalCacheManager tcm = new TransactionalCacheManager();//二级缓存
	
    public CachingExecutor(Executor delegate) {//入参是Executor的实现类对象
        this.delegate = delegate;
    }
}

下面贴一段mybatis创建执行器的源码,更好的说明CachingExecutor

public Executor newExecutor(Transaction transaction, ExecutorType 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);//默认简单执行器
    }

    //是否开启二级缓存
    if (cacheEnabled) {
    	//封装成带有二级缓存功能的执行器,在实现调用的过程中,如果开启了二级缓存,会先从缓存中获取数据,如果没有在调用具体执行器实现类,CachingExecutor也是Executor的实现类。这是什么设计模式?
        executor = new CachingExecutor(executor);
    }
    //将执行器添加到拦截器链中,在整个调用的过程中使用责任链的方式调用
    executor = (Executor) interceptorChain.pluginAll(executor);
    
    return executor;
}

二、执行流程

下面以查询为例,介绍下在开启了二级缓存时,执行器的执行流程,

1、CachingExecutor的query

@Override
public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
    //此时executor是CachingExecutor类型
    return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
    .....
}

@Override
public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
    ...
    CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql);//基于参数生成CacheKey
    return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
}

@Override
public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) {
    Cache cache = ms.getCache();
    if (cache != null) {
        //1、这里是处理二级缓存逻辑
        List<E> list = (List<E>) tcm.getObject(cache, key);
        if (list == null) {
            list = delegate.<E>query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
            tcm.putObject(cache, key, list); // issue #578 and #116
        }
        return list;
    }
    
    //2、当缓存不存在,走数据库查询,注意此时的query走的是BaseExecutor
    return delegate.<E>query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);
}

2、BaseExecutor的query

public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    //判断一级缓存是否有数据
    list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;
    if (list == null) {
        //没有,走数据查询
        list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);
    }
    return list;
}

private <E> List<E> queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {
    //注意此时的doQuery是被子类重写了,假设使用默认执行器SimpleExecutor,调用SimpleExecutor的doQuery
    list = doQuery(ms, parameter, rowBounds, resultHandler, boundSql);
    localCache.putObject(key, list);//设置一级缓存内容
    return list;
}

3、SimpleExecutor的doQuery

@Override
public <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException {
    return handler.<E>query(stmt, resultHandler);//走数据库查询
}

总结:一开始进到CachingExecutor的query方法,在调用BaseExecutor的query方法,在进到SimpleExecutor的doQuery方法查询数据库。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

信仰_273993243

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值