Spring启动扫描步骤
- 扫描@ComponentScan(“com.xx.xx”)得到路径
- 通过ResourcePatternResolver将路径下的
.class
文件都解析为Resource对象放到数组中。
Resource[] resources = ResourcePatternResolver.getResources(packageSearchPath);
- 使用ASM技术,得到类的元数据信息
MetadataReader
。它不会把类加载到JVM中。
MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource);
public final MetadataReaderFactory getMetadataReaderFactory() {
if (this.metadataReaderFactory == null) {
this.metadataReaderFactory = new CachingMetadataReaderFactory();
}
return this.metadataReaderFactory;
}
- 使用metadataReader对属性excludeFilters和includeFilters进行筛选,对@Conditional进行筛选。 includeFilters默认支持@Component注解
- 筛选通过后,基于metadataReader生成
ScannedGenericBeanDefinition
。 - 基于
ScannedGenericBeanDefinition
,判断当前处理的类如果是静态内部类、是抽象类或接口,或者不是顶级类,都不能通过筛选,其他的加入到候选beanDefinition集合。
如果抽象类中有@Lookup的方法,该类也是可以通过筛选的。 - 遍历BeanDefinition集合
7.1 给scope属性赋值
7.2 生成beanName,调用AnnotationBeanNameGenerator.generateBeanName()
7.3 解析@Lazy、@Primary、@DependsOn、@Role、@Description
等,赋值给BeanDefinition对应的属性
7.4 判断当前beanDefinition在Spring中是否已经存在,存在则报错或者跳过(重复扫描时);不存在则将beanName和BeanDefinition注册到spring容器中。 - 扫描结束。
spring扫描源码流程
- Spring容器启动时,在构造方法中,会调用
ClassPathBeanDefinitionScanner.doScan()
方法
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(Myconfig.class);
ClassPathBeanDefinitionScanner
它是扫描的核心类
postProcessBeanDefinition()
方法给beanDefinition赋默认值
processCommonDefinitionAnnotations()
方法解析@Lazy、@Primary、@DependsOn、@Role、@Description
等,赋值给BeanDefinition对应的属性
ClassPathBeanDefinitionScanner.findCandidateComponents()
会调用
ClassPathScanningCandidateComponentProvider.scanCandidateComponents()
方法,
它会去扫描注解@ComponentScan("com.xx.xx")
指定路径,Resource.getResource("path")
得到class文件集合Resource对象数组。
然后遍历resource数组,通过MetadataReader元数据读取器,把类的元信息、注解的元信息解析到metadataReader
isCandidateComponent()
找到合适的components。 默认符合spring扫描规则的是加了Component注解的class类
- 找到符合条件的类后,将它解析为
ScannedGenericBeanDefinition
,判断beanDefinition是否符合要求,是则添加到beanDefinition的集合中。
- 回到
ClassPathBeanDefinitionScanner.doScan()
方法,遍历上一步拿到的beanDefinition集合,为beanDefinition设置属性
- 检查正在注册的beanDefinition是否已存在于spring容器,存在则跳过不再注册,或者直接抛异常。
- 最后,调用
DefaultListableBeanFactory.registerBeanDefinition()
将beanDefinition注册到Spring容器中的属性beanFactory的 beanDefinitionMap集合中。它包含了synchronized同步块,所以注册beanDefinition是线程安全的。BeanDefinitionRegistry的registerBeanDefinition()实际上是调用的DefaultListableBeanFactory.registerBeanDefinition()
注册到spring容器中。