Spring bean定义的优先级是什么?

23
当多个Spring bean定义了相同的名称时,哪个会隐藏其他bean?
假设我有几个类在包org.example中使用注释@Component("bean")进行了注释,还有一个包含以下内容的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"/>

当我执行applicationContext.getBean("bean")时,将检索哪个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"/>元素。

如果您正在使用特定于测试的配置,还有哪些bean需要覆盖? - Sotirios Delimanolis
基于注解配置的bean。我的测试配置文件包括context:component-scan/,因为大多数bean都是真实的。只有少数需要被覆盖。 - Étienne Miret
启动日志有显示任何内容吗?因为正如你所说,Spring每个ID只能有一个bean。这取决于它首先加载的是注解类还是XML文件。 - Sotirios Delimanolis
除了我上面的建议之外,你们的创业日志记录有什么信息呢?它们应该输出每个创建的bean的名称。 - ninnemannk
@ninn 感谢提供 EasyMock 的链接。但是从文档中看到,它只能帮助创建模拟对象,而不能注入它们。 - Étienne Miret
显示剩余2条评论
1个回答

30
  • 在XML定义文件中,Bean的注册顺序是按照它们被找到的顺序进行的。

  • 扫描到的Bean在发现xml标签时被注册,但是扫描到的Bean不能覆盖先前注册的Bean定义。

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

所以XML获胜了。

如果将component-scan标签放在最前面,则xml beans将覆盖扫描到的Bean。如果将其放在最后,则扫描到的Bean将被忽略。

编辑

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

  • 使用别名标签声明的别名会隐藏任何后续具有相同名称的Bean定义。
  • 在名称属性中声明的别名通过抛出BeanDefinitionParsingException来防止任何其他Bean定义使用相同的名称。

例如:

 <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定义时会保存一个在同一级别的嵌套beans元素中使用的名称和别名列表,并检查名称的唯一性。

别名标签直接由DefaultBeanDefinitionDocumentReader.processAliasRegistration()处理,而解析委托对这些名称不知情。

我不知道这是否是一个错误或有意为之,但reference没有提到任何关于内部和外部别名声明具有相同行为的内容。


实际上,我注意到<alias>会覆盖任何后面的bean定义。但你说得对,无论我把component-scan元素放在哪里,扫描的bean都不能覆盖xml中的bean。 - Étienne Miret
是的,我已经测试过了。使用别名会使具有相同名称的bean定义无法访问。但是,仍然可以通过factoryBean.getBeanDefinition()检索到bean定义,但工厂不会实例化它,并且该bean将不会成为按类型自动装配的候选对象或通过factory.getBean(type)检索到。 - Jose Luis Martin
1
@JoseLuisMartin 我知道XML胜出,但如果AnnotationConfigApplicationContext扫描到两个同名的bean,它会如何决定哪个bean定义具有优先权。 - Edge

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接