需要了解的前置知识:
- Bean的生命周期
- JDK动态代理
- FactoryBean实例化
在spring的配置文件中引入下面两个bean,spring启动的时候,会加载这两个bean
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="configLocation" value="classpath:mybatis-config.xml"/>
<property name="mapperLocations" value="classpath:mapper/*.xml"/>
<property name="typeAliasesPackage" value="com.tt.spring.demo.pojo"/>
</bean>
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
<property name="basePackage" value="com.tt.spring.demo.dao"/>
</bean>
MapperScannerConfigurer实现了BeanDefinitionRegistryPostProcessor,因此spring会先进行实例化,并调用postProcessBeanDefinitionRegistry方法
在MapperScannerConfigurer类的postProcessBeanDefinitionRegistry方法中,创建了ClassPathMapperScanner对象,并调用了scan方法进行bean扫描
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {
if (this.processPropertyPlaceHolders) {
processPropertyPlaceHolders();
}
ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);
// ...省略一堆set方法
scanner.scan(
StringUtils.tokenizeToStringArray(this.basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS));
}
在ClassPathMapperScanner的doScan方法中,根据指定的basePackages,扫描指定的bean(通过重写isCandidateComponent方法,让接口可以作为beanDefinition)
protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
return beanDefinition.getMetadata().isInterface() && beanDefinition.getMetadata().isIndependent();
}
扫描完所有的beanDefinition后,调用processBeanDefinitions方法对beanDefinition进行定制,我们重点看两个地方:
private void processBeanDefinitions(Set<BeanDefinitionHolder> beanDefinitions) {
for (BeanDefinitionHolder holder : beanDefinitions) {
definition.setBeanClass(this.mapperFactoryBeanClass); // (1)
if (StringUtils.hasText(this.sqlSessionFactoryBeanName)) {
// 我们在上面定义了sqlSessionFactoryBeanName,因此会走这个分支
definition.getPropertyValues().add("sqlSessionFactory",
new RuntimeBeanReference(this.sqlSessionFactoryBeanName)); // (2)
explicitFactoryUsed = true;
} else if (this.sqlSessionFactory != null) {
definition.getPropertyValues().add("sqlSessionFactory", this.sqlSessionFactory);
explicitFactoryUsed = true;
}
}
- 代码(1)处,创建的实例实际就是mapperFactoryBeanClass对应的实例
- 代码(2)处,创建实例后会先创建sqlSessionFactoryBeanName对应的实例,然后调用setSqlSessionFactory方法
最后将这些beanDefinitions都注册到spring中
实例化:
实例化会创建MapperFactoryBean对象,它是一个FactoryBean,getObject方法的返回值才是真正的bean对象
public void setSqlSessionFactory(SqlSessionFactory sqlSessionFactory) {
if (this.sqlSessionTemplate == null || sqlSessionFactory != this.sqlSessionTemplate.getSqlSessionFactory()) {
this.sqlSessionTemplate = createSqlSessionTemplate(sqlSessionFactory);
}
}