spring中bean的作用域属性single与prototype的区别

本文深入探讨了Spring IoC容器中两种bean作用域:singleton和prototype的区别。singleton作用域下,Spring容器仅创建并返回一个共享的bean实例,适用于全局唯一标识的需求;而prototype作用域下的bean每次请求都会创建新的实例,适用于需要独立生命周期的场景。


1。当一个bean的作用域设置为singleton, 那么Spring IOC容器中只会存在一个共享的bean实例,并且所有对bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。换言之,当把一个bean定义设置为singleton作用域时,Spring IOC容器只会创建该bean定义的唯一实例。这个单一实例会被存储到单例缓存(singleton cache)中,并且所有针对该bean的后续请求和引用都将返回被缓存的对象实例,这里要注意的是singleton作用域和GOF设计模式中的单例是完全不同的,单例设计模式表示一个ClassLoader中只有一个class存在,而这里的singleton则表示一个容器对应一个bean,也就是说当一个bean被标识为singleton时候,spring的IOC容器中只会存在一个该bean。

2。prototype作用域部署的bean,每一次请求(将其注入到另一个bean中,或者以程序的方式调用容器的getBean()方法)都会产生一个新的bean实例,相当与一个new的操作,对于prototype作用域的bean,有一点非常重要,那就是Spring不能对一个prototype bean的整个生命周期负责,容器在初始化、配置、装饰或者是装配完一个prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了。不管何种作用域,容器都会调用所有对象的初始化生命周期回调方法,而对prototype而言,任何配置好的析构生命周期回调方法都将不会被调用。清除prototype作用域的对象并释放任何prototype bean所持有的昂贵资源,都是客户端代码的职责。(让Spring容器释放被singleton作用域bean占用资源的一种可行方式是,通过使用bean的后置处理器,该处理器持有要被清除的bean的引用。)

       以上可以看出,当你需要全局的唯一标示的时候可以用singleton,而且singleton只创建一个对象,系统消耗资源小.但是用singleton可能会有线程安全化的问题,这个时候就需要用到prototype 。考虑并发的问题,建议都用prototype。

虽然提供的引用未直接涉及“Spring Boot自动配置AMQP中RabbitAnnotationDrivenConfiguration类的simpleRabbitListenerContainerFactory方法参数1需要单例bean但找到2个”问题的解决方案,但可以从Spring框架的一般原理和常见处理思路来给出解决方向。 这种问题通常是由于Spring容器中存在多个符合类型要求的Bean,而方法需要单例Bean导致的。以下是一些可能的解决方案: ### 明确指定Bean名称 在使用 `@Autowired` 或方法参数注入时,明确指定要使用的Bean名称。例如,如果方法需要注入 `ConnectionFactory` 类型的Bean,而容器中有两个该类型的Bean,可以通过 `@Qualifier` 注解指定具体的Bean名称: ```java @Configuration public class SomeConfig { @Bean public SimpleRabbitListenerContainerFactory simpleRabbitListenerContainerFactory( @Qualifier("specificConnectionFactory") ConnectionFactory connectionFactory) { // 方法实现 } } ``` ### 排除不必要的自动配置 如果是自动配置引入了多余的Bean,可以通过 `@EnableAutoConfiguration` 注解的 `exclude` 属性排除相关的自动配置类。例如: ```java @SpringBootApplication(exclude = {SomeUnwantedAutoConfiguration.class}) public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` ### 调整Bean作用域 确保只有一个Bean是单例的。可以通过 `@Scope` 注解调整Bean作用域,将多余的Bean设置为非单例: ```java @Bean @Scope("prototype") public ConnectionFactory nonSingletonConnectionFactory() { // Bean创建逻辑 } ``` ### 手动配置覆盖自动配置 可以手动配置 `SimpleRabbitListenerContainerFactory`,避免使用自动配置的版本。例如: ```java @Configuration public class ManualRabbitConfig { @Bean public SimpleRabbitListenerContainerFactory mySimpleRabbitListenerContainerFactory(ConnectionFactory connectionFactory) { SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory(); factory.setConnectionFactory(connectionFactory); // 其他配置 return factory; } } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值