前言
这篇文章主要讲spring扫描的过程,讲述spring是如何通过包名去找到相关的bean。
建议搭配源码食用, 源码版本 Spring Framework 5.3.10
Spring扫描
下面是Spring扫描的源码的入口org.springframework.context.annotation.ClassPathBeanDefinitionScanner#doScan
这篇文章不会讲在哪调用这个方法, 后面Spring启动源码里会讲到
protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
Assert.notEmpty(basePackages, "At least one base package must be specified");
//结果返回值
Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();
//入参是一个包名数组,所以遍历这个数组
for (String basePackage : basePackages) {
//这个方法是找到那些被@Component注解的类, 并把它封装成一个BeanDefinition
Set<BeanDefinition> candidates = findCandidateComponents(basePackage);[关键跳转1]
//遍历刚刚的BeanDefinition,解析里面的scope属性等。
for (BeanDefinition candidate : candidates) {
//这里主要是从candidate里解析scope并放进新建的ScopeMetadata中
ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
//这里拿出刚刚解析出来的scope又设置到scope属性里
candidate.setScope(scopeMetadata.getScopeName());
//这里构造bean的名字, 先查看注解有没有指定beanName, 如果没有再通过Introspector.decapitalize生成beanName
String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);
if (candidate instanceof AbstractBeanDefinition) {
//这里会设置bean到一些初始值, 比如懒加载的值等
postProcessBeanDefinition((AbstractBeanDefinition) candidat