解析配置类
在启动Spring时,需要传入一个AppConfig.class给ApplicationContext,ApplicationContext会根据AppConfig类封装为一个BeanDefinition,这种BeanDefinition我们把它称为配置类BeanDefinition
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);
public AnnotationConfigApplicationContext(Class<?>... componentClasses) {
this();
register(componentClasses);
refresh();
}
ConfigurationClassPostProcessor是一个BeanDefinitionRegistryPostProcessor,在spring的启动流程中执行BeanFactoryBeanPostProcess的时候就会来调用这个类的postProcessBeanDefinitionRegistry方法,而里面就包含了解析配置类的逻辑,具体如下:
1.遍历所有的BeanDefinition,把配置类加到一个configCandidates集合里,排序后用于后面解析
其中分为full配置类和lite配置类两种,判断逻辑如下
2.构造一个ConfigurationClassParser用来解析配置类BeanDefinition
3.do-while循环递归解析配置类
核心方法在parser.parse(candidates)里;
a.如果配置类上存在@Component注解,那么解析配置类中的内部类(这里有递归,如果内部类也是配置类的话)
b.如果配置类上存在@PropertySource注解,那么则解析该注解,并得到PropertySource对象,并添加到environment中去
c.如果配置类上存在@ComponentScan注解,那么则解析该注解,进行扫描,扫描得到一系列的BeanDefinition对象,然后判断这些BeanDefinition是不是也是配置类BeanDefinition(只要存在@Component注解就是配置类,所以基本上扫描出来的都是配置类),如果是则递归解析该配置类,并且会生成对应的ConfigurationClass
d.如果配置类上存在@Import注解
1)如果是ImportSelector,那么调用执行selectImports方法得到类名,然后在把这个类当做配置类进行解析(也是递归)
2)如果是ImportBeanDefinitionRegistrar,那么则生成一个ImportBeanDefinitionRegistrar实例对象,并添加到配置类对象中(ConfigurationClass)的importBeanDefinitionRegistrars属性中。
e.如果配置类上存在@ImportResource注解,那么则把导入进来的资源路径存在配置类对象中的importedResources属性中。
f.如果配置类中存在@Bean的方法,那么则把这些方法封装为BeanMethod对象,并添加到配置类对象中的beanMethods属性中。
g.解析配置类所实现的接口中的@Bean,但并没有真正处理@Bean,只是暂时找出来
h.如果配置类有父类,则把父类当作配置类解析
i.AppConfig这个配置类会对应一个ConfigurationClass,同时在解析的过程中也会生成另外的一些ConfigurationClass,接下来就利用reader来进一步解析ConfigurationClass
总结
- 解析AppConfig类,生成对应的ConfigurationClass
- 再扫描,扫描到的类都会生成对应的BeanDefinition,并且同时这些类也是ConfigurationClass
- 再解析ConfigurationClass的其他信息,比如@ImportResource注解的处理,@Import注解的处理,@Bean注解的处理