本文主要介绍有些框架(MVC 框架)存在复杂的 @Bean 配置方式,如:子类覆盖父类的方法且都方法上都标注了@Bean注解,此时生效的子类的@Bean定义还是父类的@Bean定义还是两个都会生效。
1. 测试代码
作者很懒,测试代码直接拿了 Spring Boot 对 MVC 自动装配增强的一段代码。子类覆盖了父类的方法且 2 个方法都标注了@Bean注解,此时生效的是哪一个Bean呢。
我们可以自己写 2 个测试类试一试便知道了。
@Bean
public FormattingConversionService mvcConversionService() {
FormattingConversionService conversionService = new DefaultFormattingConversionService();
addFormatters(conversionService);
return conversionService;
}
@Bean
@Override
public FormattingConversionService mvcConversionService() {
Format format = this.mvcProperties.getFormat();
WebConversionService conversionService = new WebConversionService(
new DateTimeFormatters().dateFormat(format.getDate())
.timeFormat(format.getTime())
.dateTimeFormat(format.getDateTime()));
addFormatters(conversionService);
return conversionService;
}
2. 结论
子类重写了父类的方法,虽然都贴了@Bean注解但生效的是子类的Bean
3. 子类覆盖父类@Bean方法生效原理
1、ConfigurationClassParser�(ConfigurationClassPostProcessor�)解析时先解析@Bean 方法,最后才解析父类或父接口
2、子类父类的@Bean 肯定都会解析但是存放在不同位置
3、关键点在先处理子类的@Bean方法的时候,然后再处理父类的@Bean方法,而处理过程中如果发现是重写了父类的BeanDefinition�,会直接返回而不是注册BeanDefinition�,这就导致了子类的BeanDefinition生效父类的不生效。
4、代码位置org.springframework.context.annotation.ConfigurationClassBeanDefinitionReader#loadBeanDefinitionsForBeanMethod
org.springframework.context.annotation.ConfigurationClassBeanDefinitionReader#isOverriddenByExistingDefinition