社区微信群开通啦,扫一扫抢先加入社区官方微信群
社区微信群
Spring 提供了5种scope分别是:
1、singleton作用域
Spring的scope的默认值是singleton
Spring 只会为每一个bean创建一个实例,并保持bean的引用.
<bean id="bean的id" class="bean的包名.类名" scope="singleton"/> 或
<bean id="bean的id" class="bean的包名.类名" singleton="true"/>
2、prototype作用域
一个bean定义对应多个对象实例。
每一次请求(将其注入到另一个bean中,或执行getBean()方法)都会产生一个新的bean实例,相当于new操作.
Spring一旦将Bean实例交给(注入)调用者,就不再持有这个bean的引用。就无法再执行bean定义的destroy-method.
清除prototype作用域的bean对象并释放资源,是调用者的职责。
<bean id="bean的id" class="bean的包名.类名" scope="prototype"/> 或
<beanid="bean的id" class="bean的包名.类名" singleton="false"/>
3、request作用域
HTTP request表示该针对每一次HTTP请求都会产生一个新的bean,仅适用于WebApplicationContext环境。
<bean id="bean的id" class="bean的包名.类名" scope="request"/>
4、session作用域
HTTP session表示该针对每一次HTTP请求都会产生一个新的bean,仅适用于WebApplicationContext环境。
<bean id="bean的id" class="bean的包名.类名" scope="session"/>
5、globalSession作用域
在一个全局的HTTP Session中,一个bean定义对应一个实例。典型情况下,仅在使用portlet context的时候有效。该作用域仅在基于web的Spring ApplicationContext情形下有效。
考虑下面bean定义:
<bean id="bean的id" class="bean的包名.类名" scope="globalSession"/>
一个bean被声明为单例,在处理多次请求的时,在Spring 容器里只实例化出一个bean,后续的请求都公用这个对象,这个对象会保存在一个map里面。当有请求来的时候会先从缓存(map)里查看有没有,有的话直接使用这个对象,没有的话才实例化一个新的对象,所以这是个单例的。但是对于原型(prototype)bean来说当每次请求来的时候直接实例化新的bean,没有缓存以及从缓存查的过程。
1.画图分析
2.源码分析
生成bean时先判断单例的还是原型的
如果是单例的则先尝试从缓存里获取,没有在新创建
结论:
1.单例的bean只有第一次创建新的bean 后面都会复用该bean,所以不会频繁创建对象。
2.原型的bean每次都会新创建
内存方面优势明显:
1.减少了生成新实例对资源的消耗
生成新实例消耗包括两方面:第一,Spring通过反射或者cglib来生成bean实例,这都是耗性能的操作,第二,给对象分配内存会涉及复杂算法。
2.减少jvm垃圾回收
不需要给每个请求都生成新的bean实例,自然回收的对象少了。
3.可以快速获取到bean
因为单例获取bean的操作,除第一次生成外,其余都是从缓存里获取,所以很快。
有关bean实例化相关可以看着篇文章:Spring 源码分析之 bean 实例化原理
单例的bean一个很大的劣势就是他不能做到线程安全!!!,由于所有请求都共享一个bean实例,所以这个bean要是有状态的一个bean的话可能在并发场景下出现问题,而原型的bean则不会有这样问题(但也有例外,比如他被单例bean依赖),因为给每个请求都新创建实例。关于这方面我正在准备写一篇文章,在整理当中,感兴趣的朋友可以关注我,我后续写一篇详细的文章。
http://www.itxm.cn/post/14592.html
https://juejin.im/post/5cab7ebf518825177637b2f9
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!