基于SpringBoot的Mybatis源码解析:
SpringBoot版本如下:
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.5.2</version> <relativePath/> <!-- lookup parent from repository --> </parent>
Mybatis版本如下:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency> <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifactId>mybatis-spring-boot-starter</artifactId> <version>1.3.3</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency>
先来看一下引入Mybatis的效果,studentDAO实际上是一个MapperProxy对象
通过debug进去之后,可以看到sqlSession的执行:
这一切是如何发生的呢?那就得从mybatis-spring-boot-starter
的spring.factories
文件说起了。
可以看到Mybatis的自动化配置类是MybatisAutoConfiguration
。
MybatisAutoConfiguration
类中声明SqlSessionFactory
和SqlSessionTemplate
为bean
此类中的静态内部类MapperScannerRegistrarNotFoundConfiguration
引入静态内部类AutoConfiguredMapperScannerRegistrar
;
AutoConfiguredMapperScannerRegistrar
实现了ImportBeanDefinitionRegistrar
接口的 registerBeanDefinitions
方法,扫描所有Mapper
将所有Mapper
的beanDefinition
标记为MapperFactoryBean
类型的,依赖注入方式为AUTOWIRE_BY_TYPE
。
MapperFactoryBean
实现了FactoryBean#getObject()
方法,这样当创建mapper对象时(如studentDAO
)就会通过getObject()
方法来创建bean
容器的refresh()
方法里的invokeBeanFactoryPostProcessors
中,一层层下去...最后放在factoryBeanInstanceCache
里
填充属性
通过factoryBean的getObject()
方法获取代理对象:
MapperProxyFactory
代码如下: