Spring中Bean的单例和多例简单总结
在Spring中,有两个类型,一个是单例一个和多例,一个Bean同学,要么是单例要么是多例。
prototype(多例)和singleton(单例)
在Spring中,bean的Scope常被定义的两种模式:prototype(多例)和singleton(单例)。
singleton(单例):只有一个共享的实例存在,所有对这个bean的请求都会返回这个唯一的实例。
此取值时表明容器中创建时只存在一个实例,所有引用此bean都是单一实例。如同每个国家都有一个总统,国家的所有人共用此总统,而这个国家就是一个spring容器,总统就是spring创建的类的bean,国家中的人就是其它调用者,总统是一个表明其在spring中的scope为singleton,也就是单例模型。
此外,singleton类型的bean定义从容器启动到第一次被请求而实例化开始,只要容器不销毁或退出,该类型的bean的单一实例就会一直存活,典型单例模式,如同servlet在web容器中的生命周期。
prototype(多例):对这个bean的每次请求都会创建一个新的bean实例,类似于new。
spring容器在进行输出prototype的bean对象时,会每次都重新生成一个新的对象给请求方,虽然这种类型的对象的实例化以及属性设置等工作都是由容器负责的,但是只要准备完毕,并且对象实例返回给请求方之后,容器就不在拥有当前对象的引用,请求方需要自己负责当前对象后继生命周期的管理工作,包括该对象的销毁。也就是说,容器每次返回请求方该对象的一个新的实例之后,就由这个对象“自生自灭”。
如同分苹果,将苹果的bean的scope属性声明为prototype,在每个人领取苹果的时候,我们都是发一个新的苹果给他,发完之后,别人爱怎么吃就怎么吃,爱什么时候吃什么时候吃,但是注意吃完要把苹果核扔到垃圾箱!对于那些不能共享使用的对象类型,应该将其定义的scope设为prototype。
再次强调:Spring bean 默认是单例模式。
在Spring中 使用注解 @Scope进行标识一个Bean范围。其中@Scope的源码如下:
public @interface Scope {
/** * scopeName的别名 */
@AliasFor("scopeName")
String value() default "";
/**
* 默认 SCOPE_SINGLETON
* @since 4.2
* 四种类型可以选择:
* ConfigurableBeanFactory#SCOPE_PROTOTYPE :prototype
* ConfigurableBeanFactory#SCOPE_SINGLETON :singleton
* org.springframework.web.context.WebApplicationContext#SCOPE_REQUEST :request
* org.springframework.web.context.WebApplicationContext#SCOPE_SESSION :
*/
@AliasFor("value")
String scopeName() default "";
/**
* 指定组件是否应配置为作用域代理,如果是,则该代理应基于接口还是基于子类。
* 默认为ScopedProxyMode。缺省值,它通常指示,除非在组件扫描指令级别配置了不同的缺省值,否则不应该创
* 建任何作用域代理。
* 类似于Spring XML中的支持 <aop:scoped-proxy/>
*/
ScopedProxyMode proxyMode() default ScopedProxyMode.DEFAULT;
}
其中还有 request 和 session,感兴趣的朋友可以自行进行了解。
为什么用单例或者多例?何时用?
之所以用单例,是因为没必要每个请求都新建一个对象,这样子既浪费CPU又浪费内存;
之所以用多例,是为了防止并发问题;即一个请求改变了对象的状态,此时对象又处理另一个请求,而之前请求对对象状态的改变导致了对象对另一个请求做了错误的处理;
当对象含有可改变的状态时(更精确的说就是在实际应用中该状态会改变),则多例,否则单例。
如何在单例bean中注入一个多例bean
使用xml放设置Scope
使用注解方式设置Scope