Spring 注解解析核心类

Spring 注解解析核心类

1. ConfigurationClassPostProcessor

org.springframework.context.annotation.ConfigurationClassPostProcessor
这个类实现了BeanDefinitionRegistryPostProcessor接口,即在Spring 容器初始时,的invokeBeanFactoryPostProcessors方法会调用改类的postProcessBeanDefinitionRegistry方法和postProcessBeanFactory方法,这两个方法也是改类的核心方法。
在调用postProcessBeanDefinitionRegistry方法和postProcessBeanFactory方法时,他们的工作内容如下

1. postProcessBeanDefinitionRegistry
  1. 获取当前容器中所有的beanName数组并进行变量,将其对应的BeanDefinition对象获取到,然后调用ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef,
    this.metadataReaderFactory)方法,判断改类beanDefinition所对应的类是否有添加@Configuration,@Component,@ComponentScan,@Import,@ImportResource,@Bean注解,如果都有加其中任意一个注解,那么改方法会返回true。将返回true的BeanDefinition封装成一个BeanDefinitionHolder对象放到configCandidates集合中。
  2. 对canfigCandidates集合中的元素进行排序,即判断BeanDefinition对应的类中是否有加入@Order注解,如果有注解则按Order配置的值由小到大进行排序(值越小,优先及越高),如果没有则使用默认排序。
  3. 设置扫描包是的bean name生产策略,和import 的bean name生产策略,这个过程会判断当前容器中是否由beanName生成策略,如果由这将当前类的componentScanBeanNameGenerator,importBeanNameGenerator都设置为容器的,如果没有啥也不敢,通常都是啥都不做。
  4. 创建配置解析类ConfigurationClassParser对象,及两个集合candidates,alareadyParsed
  5. 进入一个do-while循环中,循环解析,其中最主要的解析步骤是调用parser.parse(candidates);这个方法,其具体的执行步骤如下
    1. 遍历configCandidates集合,判断BeanDefinition所属的类型,并根据不同的类型调用不同的方法处理,他们调用的方法名称都是叫parse(…),不过参数不一样,重载方法,不过不管该方法如何重载,最终都会调用通一个方法processConfigurationClass(ConfigurationClass configClass, Predicate filter),来进行解析操作。
    2. processConfigurationClass(…)会调用doProcessConfigurationClass(…)进行处理
    3. do_rocessConfigurationClass()方法的处理流程是
    1. 判断类上是否有加@Component注解,如果有则调用processMemberClasses(configClass, sourceClass, filter);方法解析内部内部类(即解析内部类是否有加)注解。
    2. 判断类上是否有加@PropertySources,@PropertySource注解,如果有则调用processPropertySource(propertySource);方法进行解析
    3. 判断类上是否有加@ComponentScans,@ComponentScan(会得到一个集合)注解,如果有在判是否有添加条件注解,如果没有或条件注解成功,则循环调用遍历集合,并调用this.componentScanParser.parse(componentScan, sourceClass.getMetadata().getClassName());进行解析,这个方法会返回一个集合scannedBeanDefinitions,即扫描到的BeanDefinitionHolder集合,然后在循环遍历scannedBeanDefinitions集合,调用parse(bdCand.getBeanClassName(), holder.getBeanName());方法进行解析。这是一个递归的过程。
    4. 处理@Import标签,调用processImports(configClass, sourceClass, getImports(sourceClass), filter, true);方法处理@Import注解,这里传参时有调用一个方法getImports(sourceClass),这个方法是得到该类上添加了Import注解的集合,这个方法内部会掉用collectImports(…)方法,这个方法是个递归调用的过程,他会判断当前类上的所有注解,并判断该注解内部是否有加@Import注解(递归调用)。
    5. 处理@Bean注解,首先调用retrieveBeanMethodMetadata(sourceClass);方法获取改类上加了@Bean修饰的方法集合beanMethods,然后循环调用遍历,每个方法将其封装成ConfigurationClass对象放入到configClass集合中,这个在 步骤 5中的do-while循环中会重新处理这个集合。
    6. 处理接口的默认实现processInterfaces(configClass, sourceClass);接口的默认实现是1.8提供的,所以这里会进行处理。
    7. 处理父类的注解,如果父类有添加注解,那么当前方法会返回一个SourceClass对象,在步骤 5 会执行解析这个类。
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值