这次关注的是类加载实例化部分代码。
spring中加载类在代码 ConfigurationClassPostProcessor 中实现的。
1:ConfigurationClassPostProcessor 是在AnnotationConfigEmbeddedWebApplicationContext 创建时,调用接口AnnotationConfigUtils#registerAnnotationConfigProcessors添加的bean定义,代码如下:
2:在 AbstractApplicationContext 的核心方法 refresh() 里,会调用 invokeBeanFactoryPostProcessors(beanFactory) 方法,此方法里会调用 ConfigurationClassPostProcessor 的 postProcessBeanDefinitionRegistry 方法,就是在此方法里加载了相关类。
3:postProcessBeanDefinitionRegistry 的方法里,会调用方法 processConfigBeanDefinitions,该方法的逻辑如下:
3.1:获取所有的bean定义名称,此时入口有main方法的类肯定是可以获取到。
3.2:对每一个bean名称,获取对应的bean定义信息,判断其是否有注解@Configuration而且加载过,如果有@Configuration而且没有加载过,加入列表configCandidates,并设置已经加载的属性(在 bean定义的attributes属性里,有org.springframework.context.annotation.ConfigurationClassPostProcessor.configurationClass 的key值)。代码如下:
3.3:如果有多个类有@Configuration注解,则进行排序。一般情况下,这里只有入库启动类一个。
3.4:调用ConfigurationClassParser#parse(Set<BeanDefinitionHolder>) 解析有@Configuration的类。
3.5:获取解析得到的ConfigurationClass,针对每一个ConfigurationClass,调用 ConfigurationClassBeanDefinitionReader#loadBeanDefinitions 方法处理其里面的@Import,@ImportResoure 引入加载的内容。
4:ConfigurationClassParser#parse 里的函数调用关系如下:
parse --> processConfigurationClass --> doProcessConfigurationClass。
doProcessConfigurationClass 函数的逻辑如下:
4.1:查找ConfigurationClass的内部类,如果内部类也有注解@Configuration。那么递归调用方法:processConfigurationClass。
4.2:判断类是否有注解:@ComponentScans。如此有这个注解,调用函数ComponentScanAnnotationParser#parse 处理。其处理逻辑如下:
4.2.1:解析注解@ComponentScans的属性,比如:nameGenerator,scopedProxy,includeFilters,lazyInit,basePackages等。
4.2.2:如果没有指定属性basePackages 和 basePackageClasses,则扫描路径默认为当前ConfigurationClass所在的包路径。
4.2.3:使用 PathMatchingResourcePatternResolver#getResources 扫描指定包路径下的所有class文件,并加载。
4.2.4:判断加载的类是有有注解@Component,如果有,生成bean定义对象:ScannedGenericBeanDefinition。
4.2.5:解析类上的@Scope,@Lazy, @Primary, @DependsOn等属性。
4.2.6:对扫描出来的bean定义,添加到beanFatory的缓存中。
4.3:对上一步扫描出来的bean定义,如果其有注解@Configuration,则递归调用方法:processConfigurationClass。
4.4:判断ConfigurationClass是否有@Import注解。
4.5:如果Import类实现了ImportSelector接口,则加入缓存:deferredImportSelectors。 如果Import类实现了ImportBeanDefinitionRegistrar接口,则加入:importBeanDefinitionRegistrars缓存。如果都不是,则递归调用方法:processConfigurationClass。代码:processImports(configClass, sourceClass, getImports(sourceClass), true);
4.6:判断ConfigurationClass是否有@ImportResource注解。如果有,则把locations属性指定的xml文件资源添加到缓存:importedResources。
4.7:判断ConfigurationClass的方法是否有@Bean注解。如果有,加入beanMethods缓存。
4.8:对于beanMethods缓存中的每个方法,生成ConfigurationClassBeanDefinition bean定义对象。并解析initMethod等属性,并注册到缓存。实现方法:ConfigurationClassBeanDefinitionReader#loadBeanDefinitionsForBeanMethod。
4.9:对于@ImportResoure引入的资源:importedResources。使用XmlBeanDefinitionReader.loadBeanDefinitions 方法进行解析。
4.10:处理importBeanDefinitionRegistrars缓存中的实例,调用接口ImportBeanDefinitionRegistrar的registerBeanDefinitions方法。
4.11:对于缓存deferredImportSelectors中的对象,调用其 selectImports 接口。比如:AutoConfigurationImportSelector。该类的selectImports方法里读取了META-INF/spring.factories里配置的 EnableAutoConfiguration。然后针对每个AutoConfiguration,按照ConfigurationClass进行解析处理(对于每一个AutoConfiguration, 会按照META-INF/spring-autoconfigure-metadata.properties的配置过滤一些不启动的)。比如: