一,简述 mybatis的作用就是操作数据库,其实就是封装参数,生成sql,执行sql,封装结果,其实基本就是这几个大的步骤,mybatis和spring是怎么整合的呢,以及如何一步一步执行的,具体请看下面原理分析二,源码分析2。1:SqlSessionFactoryBean 具体是在mybatisspring整合包中,SqlSessionFactoryBean是加载mybatis配置文件以及生成sqlSession的入口 2。2:FactoryBean spring源码有所了解的应该清楚这个factoryBean是一个生成特殊复杂的bean,以及和beanFactory的区:别,factoryBean中的一个重要的方法就是getObject方法,其实就是根据这个方法返回一个特殊的自定义的bean对象,而且bean创建的时候,会创建一个sqlSessionFactoryBean和sqlSessionFactory对象,就是会创建本身bean携带和getObject返回的对象,而且getObject是使用的时候才会被调用的,使用的时候才会创建这个bean交给spring来管理 调用getObejct获取sqlsessionfactory对象publicSqlSessionFactoryFactoryBean()throwsException{if(this。sqlSessionFactorynull){this。afterPropertiesSet();}returnthis。sqlSessionFactory;} 创建一个DefaultSqlSessionFactory,所以在何时创建的,就在这里 以上就是和spring结合情况下,创建出来sqlSessionFactory的过程,其实就是创建一个SqlSessionFactoryBean对象,然后调用getObject的方法,就可以做到加载mybatis的配置以及数据的封装的,以及获取一个sqlSessionFactory对象,交个spring来管理2。3:Spring中创建SqlSessionFactory对象 创建一个SqlSessionFactory,具体代码执行的逻辑如下,其实就是做了一些数据的封装,并调用getObject的方法,然后交给spring管理 2。4:springboot创建SqlSessionFactory对象MybatisAutoConfiguration BeanConditionalOnMissingBeanpublicSqlSessionFactorysqlSessionFactory(DataSourcedataSource)throwsException{还是这个对象SqlSessionFactoryBean来处理的,最后还是getObejct获取的SqlSessionFactoryBeanfactorynewSqlSessionFactoryBean();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());}SetStringfactoryPropertyNames(Set)Stream。of((newBeanWrapperImpl(SqlSessionFactoryBean。class))。getPropertyDescriptors())。map(FeatureDescriptor::getName)。collect(Collectors。toSet());Classlt;?extendsLanguageDriverdefaultLanguageDriverthis。properties。getDefaultScriptingLanguageDriver();if(factoryPropertyNames。contains(scriptingLanguageDrivers)!ObjectUtils。isEmpty(this。languageDrivers)){factory。setScriptingLanguageDrivers(this。languageDrivers);if(defaultLanguageDrivernullthis。languageDrivers。length1){defaultLanguageDriverthis。languageDrivers〔0〕。getClass();}}if(factoryPropertyNames。contains(defaultScriptingLanguageDriver)){factory。setDefaultScriptingLanguageDriver(defaultLanguageDriver);}this。applySqlSessionFactoryBeanCustomizers(factory);具体还是在这里做出来的returnfactory。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对象,根据代理对象执行相应的逻辑MapperScannerConfigurerpostProcessBeanDefinitionRegistry 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代理对象进行跟踪进行的 三,调用执行代理对象mapperProxyMapperProxyinvoke 每一个mapper调用方法的时候,就会调用invoke执行,然后调用到MapperMethod的excute方法,根据类型执行相应的增删改查 MapperMethodexecute 判断类型,其实这个类型就是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之前。