一. Spring 的 Controller 是單例還是多例?怎么保證并發(fā)的安全
答案:
controller默認(rèn)是單例的房维,不要使用非靜態(tài)的成員變量沼瘫,否則會(huì)發(fā)生數(shù)據(jù)邏輯混亂。正因?yàn)閱卫圆皇蔷€程安全的咙俩。
我們下面來簡(jiǎn)單的驗(yàn)證下:
@Controller
public class ScopeTestController {
private int num = 0;
@RequestMapping("/testScope")
public void testScope() {
System.out.println(++num);
}
@RequestMapping("/testScope2")
public void testScope2() {
System.out.println(++num);
}
}
我們首先訪問 http://localhost:8080/testScope耿戚,得到的答案是1;然后我們?cè)僭L問 http://localhost:8080/testScope2阿趁,得到的答案是 2膜蛔。
得到的不同的值,這是線程不安全的脖阵。
接下來我們?cè)賮斫ocontroller增加作用多例 @Scope("prototype")
@Controller
@Scope("prototype")
public class ScopeTestController {
private int num = 0;
@RequestMapping("/testScope")
public void testScope() {
System.out.println(++num);
}
@RequestMapping("/testScope2")
public void testScope2() {
System.out.println(++num);
}
}
我們依舊首先訪問 http://localhost:8080/testScope皂股,得到的答案是1;然后我們?cè)僭L問 http://localhost:8080/testScope2命黔,得到的答案還是 1呜呐。
相信大家不難發(fā)現(xiàn) :
單例是不安全的,會(huì)導(dǎo)致屬性重復(fù)使用纷铣。
1. 解決方案
- 不要在controller中定義成員變量卵史。
- 萬一必須要定義一個(gè)非靜態(tài)成員變量時(shí)候,則通過注解@Scope(“prototype”)搜立,將其設(shè)置為多例模式以躯。
- 在Controller中使用ThreadLocal變量
2. 補(bǔ)充說明
spring bean作用域有以下5個(gè):
singleton:單例模式,當(dāng)spring創(chuàng)建applicationContext容器的時(shí)候,spring會(huì)欲初始化所有的該作用域?qū)嵗巧瑁由蟣azy-init就可以避免預(yù)處理刁标;
-
prototype:原型模式,每次通過getBean獲取該bean就會(huì)新產(chǎn)生一個(gè)實(shí)例址晕,創(chuàng)建后spring將不再對(duì)其管理膀懈;
(下面是在web項(xiàng)目下才用到的)
request:搞web的大家都應(yīng)該明白request的域了吧,就是每次請(qǐng)求都新產(chǎn)生一個(gè)實(shí)例谨垃,和prototype不同就是創(chuàng)建后启搂,接下來的管理,spring依然在監(jiān)聽刘陶;
session:每次會(huì)話胳赌,同上;
global session:全局的web域匙隔,類似于servlet中的application疑苫。