我们知道,Executor 的基本类型有三种:SIMPLE、BATCH、REUSE,默认是SIMPLE(settingsElement()读取默认值),他们都继承了抽象类BaseExecutor。
为什么要让抽象类实现接口,然后让具体实现类继承抽象类?(模板方法模式)
“定义一个算法的骨架,并允许子类为一个或者多个步骤提供实现。
模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。”
问题:三种类型的区别(通过update()方法对比)?
SimpleExecutor:每执行一次update 或select,就开启一个Statement 对象,用完立刻关闭Statement 对象。
ReuseExecutor:执行update 或select,以sql 作为key 查找Statement 对象,存在就使用,不存在就创建,用完后,不关闭Statement 对象,而是放置于Map 内,供下一次使用。简言之,就是重复使用Statement 对象。
BatchExecutor:执行update(没有select,JDBC 批处理不支持select),将所有sql 都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个Statement 对象,每个Statement 对象都是addBatch()完毕后,等待逐一执行executeBatch()批处理。与JDBC 批处理相同。
如果配置了cacheEnabled=ture,会用装饰器模式对executor 进行包装:new CachingExecutor(executor)。
包装完毕后,会执行:
executor = (Executor) interceptorChain.pluginAll(executor);
此处会对executor 进行包装。
回答了前面的问题:数据源和事务工厂在哪里会用到——创建执行器的时候。
最终返回DefaultSqlSession,属性包括Configuration、Executor 对象。
总结:创建会话的过程,我们获得了一个DefaultSqlSession,里面包含了一个Executor,它是SQL 的执行者。