Mybatis执行流程以及整合Spring源码分析
一,简述
mybatis的作用就是操作数据库,其实就是封装参数,生成sql,执行sql,封装结果,其实基本就是这几个大的步骤,mybatis和spring是怎么整合的呢,以及如何一步一步执行的,具体请看下面原理分析
二,源码分析
2.1:SqlSessionFactoryBean
具体是在mybatis-spring整合包中,SqlSessionFactoryBean是加载mybatis配置文件以及生成sqlSession的入口

2.2:FactoryBean
spring源码有所了解的应该清楚这个factoryBean是一个生成特殊复杂的bean,以及和beanFactory的区:别,factoryBean中的一个重要的方法就是getObject方法,其实就是根据这个方法返回一个特殊的自定义的bean对象,而且bean创建的时候,会创建一个&sqlSessionFactoryBean 和sqlSessionFactory对象,就是会创建本身bean携带&和getObject返回的对象,而且getObject是使用的时候才会被调用的,使用的时候才会创建这个bean交给spring来管理

调用getObejct获取sqlsessionfactory对象
public SqlSessionFactory FactoryBean() throws Exception {
if (this.sqlSessionFactory == null) {
this.afterPropertiesSet();
}
return this.sqlSessionFactory;
}


创建一个DefaultSqlSessionFactory,所以在何时创建的,就在这里

以上就是和spring结合情况下,创建出来sqlSessionFactory的过程,其实就是创建一个SqlSessionFactoryBean对象,然后调用getObject的方法,就可以做到加载mybatis的配置以及数据的封装的,以及获取一个sqlSessionFactory对象,交个spring来管理
2.3:Spring中创建SqlSessionFactory对象
创建一个SqlSessionFactory,具体代码执行的逻辑如下,其实就是做了一些数据的封装,并调用getObject的方法,然后交给spring管理

2.4:spring-boot创建SqlSessionFactory对象
MybatisAutoConfiguration


@Bean
@ConditionalOnMissingBean
public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
//还是这个对象SqlSessionFactoryBean 来处理的,最后还是getObejct获取的
SqlSessionFactoryBean factory = new SqlSessionFactoryBean();
factory.setDataSource(dataSource);
factory.setVfs(SpringBootVFS.class);
if (StringUtils.hasText(this.properties.getConfigLocation())) {
factory.setConfigLocation(this.resourceLoader.getResource(this.properties.getConfigLocation()));
}
this.applyConfiguration(factory);
if (this.properties.getConfigurationProperties() != null) {
factory.setConfigurationProperties(this.properties.getConfigurationProperties());
}
if (!ObjectUtils.isEmpty(this.interceptors)) {
factory.setPlugins(this.interceptors);
}
if (this.databaseIdProvider != null) {
factory.setDatabaseIdProvider(this.databaseIdProvider);
}
if (StringUtils.hasLength(this.properties.getTypeAliasesPackage())) {
factory.setTypeAliasesPackage(this.properties.getTypeAliasesPackage());
}
if (this.properties.getTypeAliasesSuperType() != null) {
factory.setTypeAliasesSuperType(this.properties.getTypeAliasesSuperType());
}
if (StringUtils.hasLength(this.properties.getTypeHandlersPackage())) {
factory.setTypeHandlersPackage(this.properties.getTypeHandlersPackage());
}
if (!ObjectUtils.isEmpty(this.typeHandlers)) {
factory.setTypeHandlers(this.typeHandlers);
}
if (!ObjectUtils.isEmpty(this.properties.resolveMapperLocations())) {
factory.setMapperLocations(this.properties.resolveMapperLocations());
}
Set<String> factoryPropertyNames = (Set)Stream.of((new BeanWrapperImpl(SqlSessionFactoryBean.class)).getPropertyDescriptors()).map(FeatureDescriptor::getName).collect(Collectors.toSet());
Class<? extends LanguageDriver> defaultLanguageDriver = this.properties.getDefaultScriptingLanguageDriver();
if (factoryPropertyNames.contains("scriptingLanguageDrivers") && !ObjectUtils.isEmpty(this.languageDrivers)) {
factory.setScriptingLanguageDrivers(this.languageDrivers);
if (defaultLanguageDriver == null && this.languageDrivers.length == 1) {
defaultLanguageDriver = this.languageDrivers[0].getClass();
}
}
if (factoryPropertyNames.contains("defaultScriptingLanguageDriver")) {
factory.setDefaultScriptingLanguageDriver(defaultLanguageDriver);
}
this.applySqlSessionFactoryBeanCustomizers(factory);
//具体还是在这里做出来的
return factory.getObject();
}2.5:SqlSessionTemplate创建
MybatisAutoConfiguration中SqlSessionTemplate中创建

2.6:SqlSessionTemplate创建
创建SqlSessionTemplate 的时候,SqlSessionFactory为上文提到的DefaultSqlSessionFactory,SqlSession为这里生成的代理对象,继续根据代理对象是哪一个呢?


SqlSessionInterceptor
代理对象为这个类,看一下里面的invoke方法
invoke
代理调用
getSqlSession
获取sqlSession

openSession
获取Executor

newExecutor
InterceptorChain链执行

pluginAll
目标的增强



小结:
源码看到了这里,基本是生成了sqlSessionFactrory,SqlSessionTemplate,以及SqlSession(其实是代理),以及代理对象的invoke的执行,基本就是一些后续执行的前提,所以这些bean的创建基本都应该有所了解了
三,mappper的代理对象生成
在调用的时候,我们只是写了一个mapper接口,并未写实现,但是mapper中的方法和xml中的方法都是对应的,会将mapper全类名接口+方法名作为key存放在map中,每一个都是一个MappedStatement对象,存放在configuration的全局配置中,根据key获取到MappedStatement对象,根据代理对象执行相应的逻辑
MapperScannerConfigurer
postProcessBeanDefinitionRegistry
ClassPathMapperScanner 进行扫描,this.basePackage是扫描的包,一般就是mapper接口所在的包,springboot中是默认启动类当前包下的类被扫描 com.clover.**.mapper

scan

doScan
执行的含义就是生成bean对象

processBeanDefinitions
修改beanClass为MapperFactoryBean.class,这样创建对象的时候,就会调用MapperFactoryBean中的方法
执行afterPropertiesSet方法,然后就是往cinfiguration中添加mapper接口对象


MapperFactoryBean
实现了FactoryBean,获取mapper的时候,调用getobject方法,获取代理对象


getMapper

newInstance
获取mapper的代理对象,都是MapperProxy进行的代理,所以到此为止,就可以知道,启动的时候,已经做到了mapper的对象是从MapperProxy代理对象进行跟踪进行的

三,调用执行代理对象mapperProxy
MapperProxy
invoke
每一个mapper调用方法的时候,就会调用invoke执行,然后调用到MapperMethod的excute方法,根据类型执行相应的增删改查


MapperMethod
execute
判断类型,其实这个类型就是mapperXml定义的标签以及id和与之对应的方法名一一对应的,到了这里,可能就比较熟悉了,因为对于sql的增删改查相比都是比较的熟悉,这里其实就是下面可以猜测到的,拼装sql,和参数,然后执行sql,返回结果,处理返回结果

比如一个executeForMany为例
SqlSessionTemplate调用selectList,然后使用代理对象调用


此时的sqlsesion就是sqlSessionTemplate


代理对象为SqlSessionInterceptor,此时调用就是invoke方法

invoke方法调用
*首先获取sqlSession,然后 里面大致的流程就是新建一个DefaultSqlSession对象,并创建一个Executor的对象或者代理对象,比如pageHelper就是代理了此对象Executor
*获取Executor对象,并检查是否是需要代理
*执行代理方法
*处理一下事务的逻辑


四,mybatis的执行流程总结
执行流程图

mapperMethod执行:主要是请求参数的解析
excutor:主要是指StatementHandler的创建,包含 BoundSql 的创建、ParameterHandler和 ResultSetHandler 的创建。
statementHandler 执行:主要执行sql 并对结果集进行处理
参数组装是在excutor创建之前,拦截器是在创建excutor的时候,所以前期就是参数组装,BoundSql创建 是在StatementHandler创建的时候,在ParameterHandler和 ResultSetHandler 之前。