MyBatis二级缓存和执行流程应用
1.MyBatis基础架构
1.1 API层
提供给外部使用的API,开发人员可以通过这些API来操作数据库,API的底层执行交给数据处理层。
1.2 数据处理层
负责具体的SQL解析、执行、结果集映射。
1.3 基础组件层
负责最基础的基建,例如数据库连接管理、事务控制、缓存等。 MyBatis工作大致流产就是:从配置文件中解析得到SqlSessionFactory,然后从会话工厂中得到SqlSession,通过SqlSession去执行增删改查操作,执行完操作后关闭session会话。
2.配置文件解析
- 创建SqlSessionFactoryBuilder对象。
- 调用SqlSessionFactoryBuilder.build(InputStream is)方法。
- 在builde方法里会解析每一个配置标签或配置类参数,将信息保存在Configuration对象中。
- 解析Mapper.xml的SQL文件,将里面的每一个元素都解析出来放到一个Statement的Map中。
- 通过build(Configuration)得到一个SqlSessionFactory的实现类:DefaultSqlSessionFactory。
MyBatis的解析流程会生成一个Configuration对象,这个对象里面包含了所有的配置信息。而且解析流程最后会产生一个SqlSessionFactory,这个工厂里面也包含了这个Configuration对象。
3.MyBatis执行流程
- 通过SqlSessionFactoryBuilder解析配置文件,例如属性、别名、事务管理器、数据源等。解析完成后产生一个Configuration对象。
- 基于Configuration,再构造出SqlSessionFactory对象。
- 通过SqlSessionFactory#openSession( )方法得到一个SqlSession和一个Executor执行器,这个执行器会代理配置好的拦截器方法。
- 调用SqlSession#getMapper( )方法,得到Mapper接口的代理对象MapperProxy;接口里面所有的方法真正执行都会走到他的invoke方法。
- MapperProxy的invoke方***创建一个MapperMethod对象,然后调用MapperMethod#execute(SqlSession session ) 方法。
- 执行器Executor的#query( )方法中会创建一个StatementHandler对象,这个对象里面会创建ParameterHandler和ResultSetHandler对象。注:Executor有2个实现类:BaseExecutor和CachingExecutor。其中CachingExecutor是对BaseExecutor的装饰,它具有二级缓存的功能。
- 调用StatementHandler的CRUD方法得到结果,通过ResultSetHandler封装返回结果集。
4.MyBatis拦截器
MyBatis很多插件都是基于MyBatis的拦截器实现的。MyBatis允许在sql语句执行过程中的某一个点进行拦截调用。例如可以拦截执行器的方法、拦截参数的处理、拦截结果集、拦截SQL的构建过程等。
public interface Interceptor { Object intercept(Invocation invocation) throws Throwable; default Object plugin(Object target) { return Plugin.wrap(target, this); } default void setProperties(Properties properties) { // NOP } }举个MyBatisPlus里面的一个实现类来说:
5.MyBatis的一二级缓存
5.1 一级缓存
MyBatis的一级缓存是基于内存的,SqlSession级别的缓存,底层数据结构基于PerpetualCache的一个HashMap,key是由sql语句、条件、statement等信息组成的唯一标识,存在线程安全问题。一级缓存是强制开启的,可以通过设置statement的scope=statement来配置每次都clear缓存。一级缓存在BaseExecutor中实现。当一次SqlSession会话中,每次查询的结果集都会缓存到cache中,但是只要两次查询间,存在update、delete、insert操作就会清空缓存。
5.2 二级缓存
MyBatis的二级缓存也是基于内存,是NameSpace(Mapper)级别。默认是开启的,也可以配置关闭。二级缓存是CachingExecutor执行器管理,每当一个SqlSession在做commit操作的时候,就会清空一级缓存,将一级缓存的内容刷到二级缓存中。 一个SQL执行的时候,会先判断二级缓存中是否存在缓存,没有才会去检查一级缓存。MyBatis的二级缓可以使用软引用、弱引用、FIFO或者LRU(默认)机制来管理内存溢出问题。 二级缓存同样在遇到增删改的操作时会刷新缓存,而且二级缓存可以配置定期的刷新策略。 二级缓存默认会缓存1024个元素。
5.2.1 自定义二级缓存
MyBatis的二级缓存还可以自定义实现,通过引入ehcache的包来实现二级缓存。
#Java##计算机##编程##Java学习#