我有一个Spring bean的图表,它们互相自动装配。严重简化的插图:
<context:annotation-config/>
<bean class="Foo"/>
<bean class="Bar"/>
<bean class="Baz"/>
...
public class Foo {
@Autowired Bar bar;
@Autowired Baz baz;
}
public class Bar {
@Autowired Foo foo;
}
public class Baz {
@Autowired Foo foo;
}
所有这些bean都没有指定范围,暗示它们是单身人士(让他们明确的单身人士没有改变任何东西,我已经尝试过了。)
问题在于,在实例化单个应用程序上下文之后,Bar
和Baz
的实例包含<{>不同的实例{{1 }}。怎么会发生这种情况?
我尝试为Foo
创建public no args构造函数,并且已经确认调试已经确认Foo
不止一次。所有这些创作的堆栈跟踪都是here。
我还尝试为Spring启用调试日志记录,并在所有其他行中获得以下内容:
Foo
我理解我的bean是相互交叉引用的,但是我希望Spring框架能够尊重单例范围并初始化一个单例bean,然后将它自动装配给任何想要它的人。
有趣的事实是,如果我使用带有DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating shared instance of singleton bean 'Foo'
DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating shared instance of singleton bean 'Foo'
DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating shared instance of singleton bean 'Foo'
访问器的旧学校private
构造函数,这很好用 - 在上下文设置期间不会抛出任何异常。
FWIW,我正在使用Spring版本3.0.5(也尝试使用3.1.2,结果相同)和public static Foo getInstance
构造函数。
我可以轻松地将我的代码转换为使用静态初始化程序,但我想了解为什么Spring会以这种方式运行。这是一个错误吗?
编辑:一些额外的调查表明
o.s.c.s.ClassPathXmlApplicationContext(String ...configLocations)
的所有后续请求始终将返回context.getBean(Foo.class)
的同一实例。Foo
(大约20次使用此bean)仍会导致此对象的多个构造,但所有依赖项都使用相同的引用注入。对我来说,这表明这是一个与@Autowired
实现有关的Spring bug。我将发布到Spring社区论坛,如果我设法获得任何有用的东西,请回到这里。
答案 0 :(得分:12)
如果您不小心上下文,子上下文可以重新实例化相同的单例bean:组件扫描注释(还有其他Spring上下文扫描注释,例如MVC和其他注释)。在Web应用程序中使用Spring servlet时,这是一个常见问题,请参阅Why DispatcherServlet creates another application context?
确保您没有在子上下文中重新扫描组件,或者您只扫描特定的包/注释,并从根上下文组件扫描中排除所述包/注释。
答案 1 :(得分:1)
出于某种原因,我们也会在集成测试和服务中随机弹出这个(Spring version 4.1.4,java 1.8)。
看起来可能有不止一个罪魁祸首 - 自动装配似乎最初导致这种情况。
但是,我们通过确保为每个受影响的bean提供“id”字段来解决最一致的失败。
答案 2 :(得分:0)
尝试使用setter注入而不是构造方式,看它是否有效。在spring bean中,xml指定Bean A ref为Bean B,反之亦然。
答案 3 :(得分:0)
我的Spring配置如下:
<context:annotation-config/>
<bean class="Bar" />
<bean class="Foo" />
<bean class="Baz" />
课程与您的课程相同
测试app如下:
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class SpringTest {
/**
* @param args
*/
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("META-INF/spring/testctx.xml");
Foo foo = ctx.getBean(Foo.class);
Baz baz = ctx.getBean(Baz.class);
Bar bar = ctx.getBean(Bar.class);
System.out.println(foo.equals(baz.foo));
System.out.println(foo.equals(bar.foo));
System.out.println(baz.equals(foo.baz));
System.out.println(foo.baz.toString());
System.out.println(baz.toString());
System.out.println(foo.bar.toString());
System.out.println(bar.toString());
}
}
测试应用程序的输出如下:
true
true
true
Baz@8aef2b
Baz@8aef2b
Bar@215bf054
Bar@215bf054
使用3.0.6它完全正常(单身豆确实是单身)。可能还有其他一些你没有在这里说明搞乱你的配置。当然,作为旁注,使用默认包可能会导致一些神秘的魔法发生; - )