Spring中Bean的五個作用域
當通過spring容器創建一個Bean實例時,不僅可以完成Bean實例的實例化,還可以為Bean指定特定的作用域。Spring支持如下5種作用域:
-
singleton:單例模式,在整個Spring IoC容器中,使用singleton定義的Bean將只有一個實例
-
prototype:原型模式,每次通過容器的getBean方法獲取prototype定義的Bean時,都將產生一個新的Bean實例
-
request:對於每次HTTP請求,使用request定義的Bean都將產生一個新實例,即每次HTTP請求將會產生不同的Bean實例。只有在Web應用中使用Spring時,該作用域才有效
-
session:對於每次HTTP Session,使用session定義的Bean豆漿產生一個新實例。同樣只有在Web應用中使用Spring時,該作用域才有效
-
globalsession:每個全局的HTTP Session,使用session定義的Bean都將產生一個新實例。典型情況下,僅在使用portlet context的時候有效。同樣只有在Web應用中使用Spring時,該作用域才有效
其中比較常用的是singleton和prototype兩種作用域。對於singleton作用域的Bean,每次請求該Bean都將獲得相同的實例。容器負責跟蹤Bean實例的狀態,負責維護Bean實例的生命周期行為;如果一個Bean被設置成prototype作用域,程序每次請求該id的Bean,Spring都會新建一個Bean實例,然後返回給程序。在這種情況下,Spring容器僅僅使用new 關鍵字創建Bean實例,一旦創建成功,容器不在跟蹤實例,也不會維護Bean實例的狀態。
如果不指定Bean的作用域,Spring默認使用singleton作用域。Java在創建Java實例時,需要進行內存申請;銷毀實例時,需要完成垃圾回收,這些工作都會導致系統開銷的增加。因此,prototype作用域Bean的創建、銷毀代價比較大。而singleton作用域的Bean實例一旦創建成功,可以重復使用。因此,除非必要,否則盡量避免將Bean被設置成prototype作用域。
曾經面試的時候有面試官問我spring的controller是單例還是多例,結果
我傻逼的回答當然是多例,要不然controller類中的非靜態變量如何保證是線程安全的,這樣想起似乎是對的,但是不知道(主要是我沒看過
spring的源碼,不知道真正的內在意圖)為什麽spring的controller是單例的。
先看看spring的bean作用域有幾種,分別有啥不同。
spring bean作用域有以下5個:
singleton:單例模式,當spring創建applicationContext容器的時候,spring會欲初始化所有的該作用域實例,加上lazy-init就可以避免預處理;
prototype:原型模式,每次通過getBean獲取該bean就會新產生一個實例,創建後spring將不再對其管理;
====下面是在web項目下才用到的===
request:搞web的大家都應該明白request的域了吧,就是每次請求都新產生一個實例,和prototype不同就是創建後,接下來的管理,spring依然在監聽
session:每次會話,同上
global session:全局的web域,類似於servlet中的application
好了,上面都說了spring的controller默認是單例,那很自然就是singleton了。
再看一個例子,看看單例會不會有我說的那種問題(就是類中定義的非靜態變量線程安全問題),當然下面這個例子我是實驗過的, 要不然也不敢發出來
為什麽spring要默認是單例呢?原因有二:
1、為了性能。
2、不需要多例。
1、這個不用廢話了,單例不用每次都new,當然快了。
2、不需要實例會讓很多人迷惑,因為spring mvc官方也沒明確說不可以多例。
我這裏說不需要的原因是看開發者怎麽用了,如果你給controller中定義很多的屬性,那麽單例肯定會出現競爭訪問了。
因此,只要controller中不定義屬性,那麽單例完全是安全的。下面給個例子說明下:
默認單例的
1|2
package
com.lavasoft.demo.web.controller.lsh.ch5;
import
org.springframework.context.annotation.Scope;
import
org.springframework.stereotype.Controller;
import
org.springframework.ui.ModelMap;
import
org.springframework.web.bind.annotation.RequestMapping;
/**
* Created by Administrator on 14-4-9.
*
* @author leizhimin 14-4-9 上午10:55
*/
@Controller
@RequestMapping("/demo/lsh/ch5")
public
class MultViewController {
privateintindex =
0; //非靜態
@RequestMapping("/show")
publicStringtoShow(ModelMap model) {
System.out.println(++i);
return"/lsh/ch5/show";
}
@RequestMapping("/test")
publicStringtest() {
System.out.println(++i);
return"/lsh/ch5/test";
}
}
改為多例的(就是在class上面加一個@Scope("request")):
1 | 1
從此可見,單例是不安全的,會導致屬性重復使用。
最佳實踐:
1、不要在controller中定義成員變量。
2、萬一必須要定義一個非靜態成員變量時候,則通過註解@Scope("prototype"),將其設置為多例模式。
Spring中Bean的五個作用域