23

当定义了几个同名的 Spring bean 时,哪个会隐藏其他的?

假设我@Component("bean")在包 org.example 中有几个用注释的类,以及一个 applicationContext.xml,其中包含:

<context:component-scan base-package="org.example"/>
<alias name="aliasedBean" alias="bean"/>
<bean id="aliasedBean" class="org.example.AliasedBean"/>
<bean id="bean" class="org.example.XmlBean"/>
<import resource="otherApplicationContext.xml"/>

当我执行 ? 时将检索哪个 bean applicationContext.getBean("bean")

根据Spring 文档

每个 bean 都有一个或多个标识符。这些标识符在承载 bean 的容器中必须是唯一的。

但是,我知道(因为我测试过)当这完成时 Spring 不会抱怨。一个定义将隐藏其他定义。但我找不到规则是什么。

我想这样做是为了测试。我使用基于注释的配置来定义真正的(生产)bean。然后我想使用特定于测试的 XML 配置文件来覆盖这些定义并注入模拟 bean。

编辑:由于您有几个要求日志,我花了一些时间创建一些。他们是这样的:

  0 INFO  org.springframework.context.support.ClassPathXmlApplicationContext - Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@3934f69a: startup date [Wed Mar 06 23:04:35 CET 2013]; root of context hierarchy
 45 INFO  org.springframework.beans.factory.xml.XmlBeanDefinitionReader - Loading XML bean definitions from class path resource [applicationContext.xml]
223 INFO  org.springframework.beans.factory.support.DefaultListableBeanFactory - Overriding bean definition for bean 'bean': replacing [Generic bean: class [org.example.AnnotatedBean]; scope=singleton; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in file [/Users/etienne/Documents/Développement/Registre/workspace/SpringPrecedence/target/classes/org/example/AnnotatedBean.class]] with [Generic bean: class [org.example.XmlBean]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [applicationContext.xml]]
223 INFO  org.springframework.beans.factory.xml.XmlBeanDefinitionReader - Loading XML bean definitions from class path resource [otherApplicationContext.xml]
246 INFO  org.springframework.beans.factory.support.DefaultListableBeanFactory - Overriding bean definition for bean 'bean': replacing [Generic bean: class [org.example.XmlBean]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [applicationContext.xml]] with [Generic bean: class [org.example.ImportedXmlBean]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [otherApplicationContext.xml]]
290 INFO  org.springframework.beans.factory.support.DefaultListableBeanFactory - Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@6aba4211: defining beans [bean,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,aliasedBean,org.springframework.context.annotation.ConfigurationClassPostProcessor.importAwareProcessor]; root of factory hierarchy
290 INFO  org.example.AliasedBean - Construction of AliasedBean.
302 INFO  org.example.Main - Application context loaded.
302 INFO  org.springframework.context.support.ClassPathXmlApplicationContext - Closing org.springframework.context.support.ClassPathXmlApplicationContext@3934f69a: startup date [Wed Mar 06 23:04:35 CET 2013]; root of context hierarchy
302 INFO  org.springframework.beans.factory.support.DefaultListableBeanFactory - Destroying singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@6aba4211: defining beans [bean,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,aliasedBean,org.springframework.context.annotation.ConfigurationClassPostProcessor.importAwareProcessor]; root of factory hierarchy

经过一些测试,我发现在创建上下文时会出现异常,如果:

  • 我有两个@Component("bean")
  • 我在同一个XML 文件中有两个<bean id="bean"/>元素。
4

1 回答 1

29
  • bean 按照在 xml 定义文件中找到的顺序注册。

  • 扫描的 bean 在找到 xml 标记时注册,但扫描的 bean 不能覆盖以前注册的 bean 定义。

  • DefaultListableBeanFactory.allowBeanDefinitionOverriding如果为 true(默认情况下),XML bean 定义可以覆盖任何以前的 bean 定义。

所以 XML 获胜。

如果您首先放置组件扫描标记,xml bean 将覆盖扫描的那些。如果你把它放在最后,扫描的 bean 将被忽略。

编辑

如果在 bean 定义中的 name 属性中声明或使用别名标记声明,则别名具有不同的行为。

  • 使用 alias 标记声明的 aliases 隐藏了任何以后具有相同名称的 bean 定义。
  • 在 name 属性中声明的别名通过抛出BeanDefinitionParsingException.

例如:

 <bean id="foo" name="bar" class="Foo" />
 <bean id="bar" class="Bar" />   -- throw Exception (name bar is in use)

<bean id="foo" class="Foo" />
<alias name="foo" alias="bar" />
<bean id="bar" class="Bar" /> -- Hidden by alias no exception thrown

不同之处在于,BeanDefinitionParserDelegate在 bean 元素嵌套的同一 bean 级别保存名称和别名列表,并在解析 bean 定义时检查名称的唯一性。

别名标记由直接处理DefaultBeanDefinitionDocumentReader.processAliasRegistration(),解析器委托不知道此名称。

I don't know if it's a bug or intentional but reference does not say anything about and seem to be expected that the internal and external declarations of aliases have the same behavior.

于 2013-03-05T23:49:00.307 回答