概 述
熟悉 Spring开发的朋友都知道 Spring 提供了 5种 scope
,分别是:
-
singleton
: 单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理; -
prototype
:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;
(下面是在web项目下才用到的) -
request
:搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听; -
session
: 每次会话,同上; -
global session
: 全局的web域,类似于servlet中的application。
如下图是官方文档上的截图,感兴趣的朋友可以进去看看这五种分别有什么不同。今天要介绍的是这五种中的前两种,也是 Spring
最初提供的 bean scope
:singleton
和 prototype
Spring 官方文档介绍如下图:
单例bean与原型bean的区别
如果一个 bean
被声明为单例的时候,在处理多次请求的时候在 Spring
容器里只实例化出一个bean
,后续的请求都公用这个对象,这个对象会保存在一个map
里面。当有请求来的时候会先从 缓存(map
) 里查看有没有,有的话直接使用这个对象,没有的话才实例化一个新的对象,所以这是个单例的。但是对于 原型(prototype
) bean
来说当每次请求来的时候直接实例化新的bean
,没有缓存以及从缓存查的过程。
1、画图分析
2、源码分析
生成 bean时先判断单例的还是原型的
如果是单例的则先尝试从缓存里获取,没有在新创建
3、结论:
- 单例的 bean只有第一次创建新的 bean 后面都会复用该 bean,所以不会频繁创建对象。
- 原型的 bean每次都会新创建
单例bean的优势
由于不会每次都新创建新对象所以有一下几个性能上的优势:
- 减少了新生成实例的消耗 新生成实例消耗包括两方面,第一,Spring会通过反射或者 cglib来生成 bean实例,这都是耗性能的操作,其次给对象分配内存也会涉及复杂算法
- 减少jvm垃圾回收 由于不会给每个请求都新生成bean实例,所以自然回收的对象少了
- 可以快速获取到 bean 因为单例的获取bean操作除了第一次生成之外其余的都是从缓存里获取的,所以很快
单例bean的劣势
单例的 bean一个很大的劣势就是他不能做到线程安全!!!由于所有请求都共享一个 bean实例,所以这个 bean要是有状态的一个 bean的话可能在并发场景下出现问题,而原型的 bean则不会有这样问题(但也有例外,比如他被单例 bean依赖),因为给每个请求都新创建实例。