Controller是單例還是多例?
controller默認是單例的,不要使用非靜態的成員變量,否則會發生數據邏輯混亂。正因為單例所以不是線程安全的。
我們下面來簡單的驗證下:
package com.riemann.springbootdemo.controller;import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;@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。
得到的不同的值,這是線程不安全的。
接下來我們再來給controller增加作用多例 @Scope(“prototype”)
package com.riemann.springbootdemo.controller;import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;@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中定義成員變量。
萬一必須要定義一個非靜態成員變量時候,則通過注解@Scope(“prototype”),將其設置為多例模式。
在Controller中使用ThreadLocal變量
補充說明
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。