添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
相关文章推荐
卖萌的烤地瓜  ·  jquery ...·  4 月前    · 
乖乖的李子  ·  Linux : nano: command ...·  1 年前    · 
粗眉毛的桔子  ·  docker ...·  1 年前    · 
本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《 阿里云开发者社区用户服务协议 》和 《 阿里云开发者社区知识产权保护指引 》。如果您发现本社区中有涉嫌抄袭的内容,填写 侵权投诉表单 进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

spring 的bean默认是单例,这个用spring的人基本都知道。

如果需要多个实例,又要使用ioc怎么办呢?

当然是使用@Scope注解,指明ConfigurableBeanFactory.SCOPE_PROTOTYPE了。

* Scope identifier for the standard singleton scope: "singleton". * Custom scopes can be added via {@code registerScope}. * @see #registerScope String SCOPE_SINGLETON = "singleton"; * Scope identifier for the standard prototype scope: "prototype". * Custom scopes can be added via {@code registerScope}. * @see #registerScope String SCOPE_PROTOTYPE = "prototype";

但是在使用过程中 发现这样依然是单例。

难道spring的注解不稳定还是我使用有误?

首先推敲一下,为什么我们获取到的实例每次都一样。

spring出问题是不太可能了,这么常见的特性那么多人都用呢。

如果给一个组件加上

@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)

每次请求它的实例,spring的确会给返回一个新的。

问题是这个多例对象是被其他单例服务依赖的。

而单例服务初始化的时候,多例对象就已经被创建好了。当你去使用单例服务的时候,多例对象也不会被再次创建了。

那怎么解决呢?

一个首先想到的方法当然是注入多个,就是写多个@Autowired。

这样的确可以。而且如果对单例服务写多个,他们也是一样的,只有对多例的才行。

但是我们的服务绝大多数时候都不知道需要多少多例服务,服务是动态创建的。

所以另一种方法就是使用getBean方法:

@Autowired
private ApplicationContext applicationContext;
applicationContext.getBean()

只要每次调用getBean就可以获得一个新服务。

其实spring的@Scope注解提供了在单例服务里使用多例对象的能力,它提供了一个代理字段

* Specifies whether a component should be configured as a scoped proxy * and if so, whether the proxy should be interface-based or subclass-based. * <p>Defaults to {@link ScopedProxyMode#DEFAULT}, which typically indicates * that no scoped proxy should be created unless a different default * has been configured at the component-scan instruction level. * <p>Analogous to {@code <aop:scoped-proxy/>} support in Spring XML. * @see ScopedProxyMode ScopedProxyMode proxyMode() default ScopedProxyMode.DEFAULT;

默认是不使用代理创建。我们只要把它改成使用代理即可:

@Component
@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE, proxyMode = ScopedProxyMode.TARGET_CLASS)

 这样即使是在单例里面使用,每次获取多例对象也会拿到一个新的。