Spring中bean的作用域scope详解

2022-12-30,,,,

参考文献:http://blog.csdn.net/jacklearntech/article/details/40157861 
http://www.cnblogs.com/qq78292959/p/3716827.html
网上查找的资料都是说三种scope类型request,session、globalSession,而我在测试跟踪源码过程中却发现的是四种。。。。。。

scope用来声明IOC容器中的对象应该处的限定场景或者说该对象的存活空间,即在IOC容器在 对象进入相应的scope之前,生成并装配这些对象,在该对象不再处于这些scope的限定之后,容器通常会销毁这些对象。打个比方吧!我们都是处在社会 (容器)中,如果把中学教师作为一个类定义,那么当容器初始化这些类之后,中学教师只能局限在中学这个场景中,中学,就可以看做中学教师的scope。

Spring容器最初提供了两种bean的scope类型:singleton和 prototype,但发布2.0之后,又引入了另外四种scope类型,即request,session、globalSession和application类型。不过这四种scope类型只能在web应用中使用,在WebApplicationContext类中指定了支持的scope类型。

注意:这里的singleton和设计模式里面的单例模式不一样,标记为singleton的bean是由容器来保证这种类型的bean在同一个容器内只存在一个共享实例,而单例模式则是保证在同一个Classloader中只存在一个这种类型的实例。

1、singleton

singleton类型的bean定义,在一个容器中只存在一个实例,所有对该类型bean的依赖都引用这一单一实例,这就好像每个幼儿园都会有一个滑梯一样,这个幼儿园的小朋友共同使用这一个滑梯,而对于幼儿园容器来说,滑梯就是一个singleton的bean。

此外,singleton类型的bean定义,从容器启动,到他第一次被请求而实例化开始,只要容器不销毁或退出,该类型的bean的单一实例就会一直存活。

通常情况下,如果你不指定bean的scope,singleton便是容器默认的scope,所以,下面三种配置,形式实际上达成的是同样的效果:

2、prototype

scope为prototype的bean,容器在接受到该类型的对象的请求的时候,会每次都重新 生成一个新的对象给请求方,虽然这种类型的对象的实例化以及属性设置等工作都是由容器负责的,但是只要准备完毕,并且对象实例返回给请求方之后,容器就不 在拥有当前对象的引用,请求方需要自己负责当前对象后继生命周期的管理工作,包括该对象的销毁。也就是说,容器每次返回请求方该对象的一个新的实例之后, 就由这个对象“自生自灭”了。

让我们继续幼儿园的比喻,我们今天要分苹果了!将苹果的bean的scope属性声明为 prototype,在每个小朋友领取苹果的时候,我们都是发一个新的苹果给他,发完之后,小朋友爱怎么吃就怎么吃,爱什么时候吃什么时候吃,但是注意吃 完要把果核扔到垃圾箱哦!对于那些不能共享使用的对象类型,应该将其定义的scope设为prototype,通常,声明为prototype的的 bean,都是一些有状态的,比如保存为每个顾客信息的对象。

3、request 、session、globalSession和application
spring在web容器启动时就会往bean工厂里面注册对应域处理类,在WebApplicationContextUtils类registerWebApplicationScopes方法中注册

request:spring容器会将scope="request"的bean放到HttpServletRequest.setAttribute(name, value)中,bean的生命周期同HttpServletRequest。

session、globalSession:经测试这两种都是同样的功能控制,spring容器会将scope="session/globalSession"的bean放到HttpSession.setAttribute(name, value)中,bean生命周期同HttpSession。

使用request、session、globalSession三种scope类型,需要在web.xml中添加监听器RequestContextListener/过滤器RequestContextFilter(使用哪种与servlet版本相关),spring在拦截到请求后会将相关的请求对象信息储存起来,供spring工厂容器创建bean时使用。(注:这三种域的类不建议作为其他域的引用或注入对象,如在容器启动创建bean时这三个域的存储器并不存在,容易出现异常)
以RequestContextListener为例,spring会将request封装后保存到RequestContextHolder中,RequestContextHolder将对象保存到ThreadLocal类型属性中(按请求线程区分存储),如下图

application:spring容器会将scope="application"的bean放到ServletContext.setAttribute(name, value)中,生命周期同ServletContext。

Spring中bean的作用域scope详解的相关教程结束。

《Spring中bean的作用域scope详解.doc》

下载本文的Word格式文档,以方便收藏与打印。