1、Mybatis执行流程
第一步:通过Resource加载配置好Mybatis.xml配置文件
调用 ibatis.io包下面的 Resources类中的getResourceAsStream()方法,把xml文件解析为一个流;
源码:
第二步:new 一个SqlSessionFactoryBuilder对象,他是SqlSessionFactory的构建者,调用他的build()方法构建SqlSessionFactory
源码:
这里的 XMLconfigBuilder对象 是用来解析XML文件的一个构造者,通过他的parse()方法解析Mybatis.xml配置文件
解析Configuration节点下的子节点:
parse()解析完成后,返回一个configuration对象
返回来的 configuration 作为参数传入另一个build()方法,然后又返回了一个DefaultSqlSessionFactory对象
他是SqlSessionFactory的一个实现类,用来生产defaultSqlSession对象的;到这里,第二步就完成了。
既然DefaultSqlSessionFactory是SqlSessionFactory的实现类,为什么我们不直接new一个defaultSqlSessionFactory呢?
原因:看源码知道,如果new defaultSqlSessionFactory的话,我们需要一个configuration对象,而这个对象有事经过层层磨难才得到的,所以我们一般采取构建筑模式,直接把xml字节流或字符流给了sqlSessionFactoryBuilder
第三步: 现在我们有了一个SqlSessionFactory了 , 我们最终的目的是获取一个SqlSession对象 ,这里 调用了openSession()方法
里面有一个Transaction,这是一个事务,transaction产生需要的参数是environment,而environment正是从configuration(xml中配置的)获得的,通过这些参数transactionFactory就产生了transaction。
executor是mybatis核心执行器,发送sql语句并执行 , 相当于jdbc中的statement,有了这些,最后通过DefaultSqlSession构造方法得到sqlSession。
第四步: jdk动态代理生成mapper接口的代理对象
通过DefaultSqlSession的getMapper()来生成
进入configuration.getMapper()方法:
再进入mapperRegistry.getMapper()方法:
其实是通过mapperRegistry的对象,里面是一个 knownMappers的HashMap , 用于存储配置文件中的mappers标签的每个mapper接口
这里通过JDK动态代理,帮mapper接口生成了代理实现类
第五步:通过第四步返回的代理对象的getUserList()方法调用getMapper方法最终执行的方法
代理对象的getUser方法执行其实走的是MapperProxy的invoke方法
执行器: 这里是判断sql的执行类型
查询方法
走executor的query方法
走CachingExecutor的query方法,先从二级缓存中获取
如二级缓存中没有数据,走delegate(BaseExecutor)的query方法 ,也就是一级缓存localCache,如果一级缓存没有数据,则走queryFromDatabase方法查数据库
从数据库查到数据,放入到一级缓存中
一级缓存底层是个PerpetualCache类型的HashMap
总结:mybatis运行时首先通过Resource.getResourceAsStream()方法把mybatis的核心配置文件加载进来,然后通过XMLConfigBuilder的parse()方法解析完成后把结果放入 configuration中,以configuration为参数传入build()方法并返回一个defaultSqlSessionFactory。然后我们调用openSession()方法来获取SqlSession,在构建SqlSession的同时还需要transaction和executor用于后续执行操作。