一、驗證controller線程安全
controller默認(rèn)是單例的,不要使用非靜態(tài)的成員變量,否則會發(fā)生數(shù)據(jù)邏輯混亂。正因為單例所以不是線程安全的。
我們下面來簡單的驗證下:
@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;然后我們再訪問 http://localhost:8080/testScope2,得到的答案是2。
得到的不同的值,單例會導(dǎo)致屬性重復(fù)使用,這是線程不安全的。
二、controller多例
接下來我們再來給controller增加作用多例注解@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;然后我們再訪問 http://localhost:8080/testScope2,得到的答案還是1。
三、解決方案
- 不要在controller中定義成員變量。
- 萬一必須要定義一個非靜態(tài)成員變量時候,則通過注解@Scope(“prototype”),將其設(shè)置為多例模式。
- 在controller中使用ThreadLocal變量。
四、作用域補充說明
spring bean作用域有以下5個:
- singleton:單例模式,當(dāng)spring創(chuàng)建applicationContext容器的時候,spring會欲初始化所有的該作用域?qū)嵗?,加上lazy-init就可以避免預(yù)處理;
- prototype:原型模式,每次通過getBean獲取該bean就會新產(chǎn)生一個實例,創(chuàng)建后spring將不再對其管理;
------(下面是在web項目下才用到的)------ - request:搞web的大家都應(yīng)該明白request的域了吧,就是每次請求都新產(chǎn)生一個實例,和prototype不同就是創(chuàng)建后,接下來的管理,spring依然在監(jiān)聽;
- session:每次會話,同上;
- global session:全局的web域,類似于servlet中的application。